Skip to content
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

Finish unfinished Spans on Transaction completion #1296

Merged
merged 22 commits into from
Nov 8, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
22 commits
Select commit Hold shift + click to select a range
c2d186b
add test case.
lucas-zimerman Nov 2, 2021
872c2c1
Update test.
lucas-zimerman Nov 2, 2021
e2a32d1
removed some moqs
lucas-zimerman Nov 2, 2021
5591695
Code cleanup
lucas-zimerman Nov 2, 2021
34c498a
Close spans with events without exceptions.
lucas-zimerman Nov 3, 2021
ae96241
Format code
getsentry-bot Nov 3, 2021
cfb2519
rollback Hub CaptureEvent, removed `A` and implemented close unfinish…
lucas-zimerman Nov 4, 2021
9d3ddeb
typo.
lucas-zimerman Nov 4, 2021
d04bf92
Tests fixes.
lucas-zimerman Nov 4, 2021
8038daa
Merge branch 'main' into fix/message-only-errors-not-closing-span
lucas-zimerman Nov 4, 2021
657c9b5
small refactor
lucas-zimerman Nov 5, 2021
3c943fd
Merge branch 'fix/message-only-errors-not-closing-span' of https://gi…
lucas-zimerman Nov 5, 2021
0bd5097
Merge branch 'main' into fix/message-only-errors-not-closing-span
lucas-zimerman Nov 5, 2021
272eece
Update changelog.
lucas-zimerman Nov 5, 2021
07374b2
Self code review fixes.
lucas-zimerman Nov 5, 2021
f30239b
Moved OR condition to the left side.
lucas-zimerman Nov 5, 2021
a92e53c
Removed unneeded Span close.
lucas-zimerman Nov 8, 2021
1223d3d
Merge branch 'main' into fix/message-only-errors-not-closing-span
lucas-zimerman Nov 8, 2021
dc2895a
Format code
getsentry-bot Nov 8, 2021
bb0063e
Merge branch 'main' into fix/message-only-errors-not-closing-span
lucas-zimerman Nov 8, 2021
fa2aa9e
Removed unused extension.
lucas-zimerman Nov 8, 2021
b25266a
Merge branch 'fix/message-only-errors-not-closing-span' of https://gi…
lucas-zimerman Nov 8, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
# Changelog

## Unreleased

### Fixes

- Finish unfinished Spans on Transaction completion ([#1296](https://github.com/getsentry/sentry-dotnet/pull/1296))

## 3.12.0-alpha.1

### Features
Expand Down
5 changes: 0 additions & 5 deletions src/Sentry/Internal/Hub.cs
Original file line number Diff line number Diff line change
Expand Up @@ -285,11 +285,6 @@ public SentryId CaptureEvent(SentryEvent evt, Scope? scope = null)
evt.Contexts.Trace.TraceId = linkedSpan.TraceId;
evt.Contexts.Trace.ParentSpanId = linkedSpan.ParentSpanId;
}
else if (evt.IsErrored() && scope?.LastCreatedSpan() is { } lastSpan && lastSpan?.IsFinished == false)
{
// Can still be reset by the owner but lets consider it finished and errored for now.
lastSpan.Finish(SpanStatus.InternalError);
}

actualScope.SessionUpdate = evt switch
{
Expand Down
13 changes: 0 additions & 13 deletions src/Sentry/SentryEventExtensions.cs

This file was deleted.

2 changes: 1 addition & 1 deletion src/Sentry/Transaction.cs
Original file line number Diff line number Diff line change
Expand Up @@ -225,7 +225,7 @@ public Transaction(ITransaction tracer)
_breadcrumbs = tracer.Breadcrumbs.ToList();
_extra = tracer.Extra.ToDictionary();
_tags = tracer.Tags.ToDictionary();
_spans = tracer.Spans.Where(s => s.IsFinished).Select(s => new Span(s)).ToArray();
_spans = tracer.Spans.Select(s => new Span(s)).ToArray();
lucas-zimerman marked this conversation as resolved.
Show resolved Hide resolved
}

/// <inheritdoc />
Expand Down
8 changes: 8 additions & 0 deletions src/Sentry/TransactionTracer.cs
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,14 @@ public void Finish()
Status ??= SpanStatus.UnknownError;
EndTimestamp = DateTimeOffset.UtcNow;

foreach (var span in _spans)
{
if (!span.IsFinished)
{
span.Finish(SpanStatus.DeadlineExceeded);
}
}

// Clear the transaction from the scope
_hub.ConfigureScope(scope => scope.ResetTransaction(this));

Expand Down
48 changes: 44 additions & 4 deletions test/Sentry.Tests/HubTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -277,26 +277,29 @@ public void CaptureEvent_SessionActive_NoExceptionDoesNotReportError()
}

[Fact]
public void CaptureEvent_ExceptionWithOpenSpan_SpanFinishedWithInternalError()
public void CaptureEvent_ExceptionWithOpenSpan_SpanLinkedToEventContext()
{
// Arrange
var client = Substitute.For<ISentryClient>();

var hub = new Hub(new SentryOptions
{
Dsn = DsnSamples.ValidDsnWithSecret,
TracesSampleRate = 1
}, client);
var scope = new Scope();
var evt = new SentryEvent(new Exception());
scope.Transaction = hub.StartTransaction("transaction", "operation");

var child = scope.Transaction.StartChild("child", "child");

// Act
hub.CaptureEvent(new SentryEvent(new Exception()), scope);
hub.CaptureEvent(evt, scope);

// Assert
Assert.Equal(SpanStatus.InternalError, child.Status);
Assert.True(child.IsFinished);
Assert.Equal(child.SpanId, evt.Contexts.Trace.SpanId);
Assert.Equal(child.TraceId, evt.Contexts.Trace.TraceId);
Assert.Equal(child.ParentSpanId, evt.Contexts.Trace.ParentSpanId);
}

[Fact]
Expand Down Expand Up @@ -1016,5 +1019,42 @@ public void ResumeSession_NoPausedSession_DoesNothing()
// Assert
client.DidNotReceive().CaptureSession(Arg.Is<SessionUpdate>(s => s.EndStatus != null));
}

[Theory]
[InlineData(SentryLevel.Warning)]
[InlineData(SentryLevel.Info)]
[InlineData(SentryLevel.Debug)]
[InlineData(SentryLevel.Error)]
[InlineData(SentryLevel.Fatal)]
public void CaptureEvent_MessageOnlyEvent_SpanLinkedToEventContext(SentryLevel level)
{
// Arrange
var client = Substitute.For<ISentryClient>();

var hub = new Hub(new SentryOptions
{
Dsn = DsnSamples.ValidDsnWithSecret,
TracesSampleRate = 1
}, client);
var scope = new Scope();
var evt = new SentryEvent()
{
Message = "Logger error",
Level = level
};
scope.Transaction = hub.StartTransaction("transaction", "operation");

var child = scope.Transaction.StartChild("child", "child");

// Act
hub.CaptureEvent(evt, scope);

// Assert
Assert.Equal(child.SpanId, evt.Contexts.Trace.SpanId);
Assert.Equal(child.TraceId, evt.Contexts.Trace.TraceId);
Assert.Equal(child.ParentSpanId, evt.Contexts.Trace.ParentSpanId);
Assert.False(child.IsFinished);
Assert.Null(child.Status);
}
}
}
37 changes: 17 additions & 20 deletions test/Sentry.Tests/Protocol/TransactionTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -235,46 +235,43 @@ public void Finish_RecordsTime()
}

[Fact]
public void Finish_CapturesTransaction()
public void Finish_UnfinishedSpansGetsFinishedWithDeadlineStatus()
{
// Arrange
var client = Substitute.For<ISentryClient>();
var options = new SentryOptions { Dsn = DsnSamples.ValidDsnWithoutSecret };
var hub = new Hub(options, client);

var transaction = new TransactionTracer(hub, "my name", "my op");
var transaction = new TransactionTracer(DisabledHub.Instance, "my name", "my op");
transaction.StartChild("children1");
transaction.StartChild("children2");
transaction.StartChild("children3.finished").Finish(SpanStatus.Ok);
transaction.StartChild("children4");

// Act
transaction.Finish();

// Assert
client.Received(1).CaptureTransaction(Arg.Any<Transaction>());

Assert.All(transaction.Spans.Where(span => !span.Operation.EndsWith("finished")), span =>
{
Assert.True(span.IsFinished);
Assert.Equal(SpanStatus.DeadlineExceeded, span.Status);
});
Assert.Single(transaction.Spans.Where(span => span.Operation.EndsWith("finished") && span.Status == SpanStatus.Ok));
}

[Fact]
public void Finish_DropsUnfinishedSpans()
public void Finish_CapturesTransaction()
{
// Arrange
var client = Substitute.For<ISentryClient>();
var hub = new Hub(new SentryOptions { Dsn = DsnSamples.ValidDsnWithoutSecret }, client);
var options = new SentryOptions { Dsn = DsnSamples.ValidDsnWithoutSecret };
var hub = new Hub(options, client);

var transaction = new TransactionTracer(hub, "my name", "my op");

transaction.StartChild("op1").Finish();
transaction.StartChild("op2");
transaction.StartChild("op3").Finish();

// Act
transaction.Finish();

// Assert
client.Received(1).CaptureTransaction(
Arg.Is<Transaction>(t =>
t.Spans.Count == 2 &&
t.Spans.Any(s => s.Operation == "op1") &&
t.Spans.All(s => s.Operation != "op2") &&
t.Spans.Any(s => s.Operation == "op3")
));
client.Received(1).CaptureTransaction(Arg.Any<Transaction>());
}

[Fact]
Expand Down