Skip to content

Commit e607d46

Browse files
Address usecase for cancelled cancellation token
1 parent 632fa98 commit e607d46

File tree

3 files changed

+49
-13
lines changed

3 files changed

+49
-13
lines changed

src/Microsoft.Data.SqlClient/netcore/src/Microsoft/Data/SqlClient/SqlDataReader.cs

+7-6
Original file line numberDiff line numberDiff line change
@@ -4733,18 +4733,19 @@ public override Task<bool> ReadAsync(CancellationToken cancellationToken)
47334733
return Task.FromException<bool>(ADP.ExceptionWithStackTrace(ADP.DataReaderClosed()));
47344734
}
47354735

4736-
// If user's token is canceled, return a canceled task
4737-
if (cancellationToken.IsCancellationRequested)
4738-
{
4739-
return Task.FromCanceled<bool>(cancellationToken);
4740-
}
4741-
4736+
// Register first to catch any already expired tokens to be able to trigger cancellation event.
47424737
IDisposable registration = null;
47434738
if (cancellationToken.CanBeCanceled)
47444739
{
47454740
registration = cancellationToken.Register(SqlCommand.s_cancelIgnoreFailure, _command);
47464741
}
47474742

4743+
// If user's token is canceled, return a canceled task
4744+
if (cancellationToken.IsCancellationRequested)
4745+
{
4746+
return Task.FromCanceled<bool>(cancellationToken);
4747+
}
4748+
47484749
// Check for existing async
47494750
if (_currentTask != null)
47504751
{

src/Microsoft.Data.SqlClient/netfx/src/Microsoft/Data/SqlClient/SqlDataReader.cs

+7-6
Original file line numberDiff line numberDiff line change
@@ -5326,18 +5326,19 @@ public override Task<bool> ReadAsync(CancellationToken cancellationToken)
53265326
return ADP.CreatedTaskWithException<bool>(ADP.ExceptionWithStackTrace(ADP.DataReaderClosed("ReadAsync")));
53275327
}
53285328

5329-
// If user's token is canceled, return a canceled task
5330-
if (cancellationToken.IsCancellationRequested)
5331-
{
5332-
return ADP.CreatedTaskWithCancellation<bool>();
5333-
}
5334-
5329+
// Register first to catch any already expired tokens to be able to trigger cancellation event.
53355330
IDisposable registration = null;
53365331
if (cancellationToken.CanBeCanceled)
53375332
{
53385333
registration = cancellationToken.Register(SqlCommand.s_cancelIgnoreFailure, _command);
53395334
}
53405335

5336+
// If user's token is canceled, return a canceled task
5337+
if (cancellationToken.IsCancellationRequested)
5338+
{
5339+
return ADP.CreatedTaskWithCancellation<bool>();
5340+
}
5341+
53415342
// Check for existing async
53425343
if (_currentTask != null)
53435344
{

src/Microsoft.Data.SqlClient/tests/ManualTests/SQL/DataReaderTest/DataReaderCancellationTest.cs

+35-1
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,41 @@ await Assert.ThrowsAsync<TaskCanceledException>(async () =>
4242
}
4343
}
4444
});
45-
Assert.True(stopwatch.ElapsedMilliseconds < 1000, "Cancellation did not trigger on time.");
45+
Assert.True(stopwatch.ElapsedMilliseconds < 10000, "Cancellation did not trigger on time.");
46+
}
47+
}
48+
49+
/// <summary>
50+
/// Test ensures cancellation token is registered before ReadAsync starts processing results from TDS Stream,
51+
/// such that when Cancel is triggered, the token is capable of canceling reading further results.
52+
/// </summary>
53+
/// <returns>Async Task</returns>
54+
[ConditionalFact(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringsSetup))]
55+
public static async Task CancelledCancellationTokenIsRespected_ReadAsync()
56+
{
57+
const string longRunningQuery = @"
58+
with TenRows as (select Value from (values (1), (2), (3), (4), (5), (6), (7), (8), (9), (10)) as TenRows (Value)),
59+
ThousandRows as (select A.Value as A, B.Value as B, C.Value as C from TenRows as A, TenRows as B, TenRows as C)
60+
select *
61+
from ThousandRows as A, ThousandRows as B, ThousandRows as C;";
62+
63+
using (var source = new CancellationTokenSource())
64+
using (var connection = new SqlConnection(DataTestUtility.TCPConnectionString))
65+
{
66+
await connection.OpenAsync(source.Token);
67+
68+
Stopwatch stopwatch = Stopwatch.StartNew();
69+
await Assert.ThrowsAsync<TaskCanceledException>(async () =>
70+
{
71+
using (var command = new SqlCommand(longRunningQuery, connection))
72+
using (var reader = await command.ExecuteReaderAsync(source.Token))
73+
{
74+
source.Cancel();
75+
while (await reader.ReadAsync(source.Token))
76+
{ }
77+
}
78+
});
79+
Assert.True(stopwatch.ElapsedMilliseconds < 10000, "Cancellation did not trigger on time.");
4680
}
4781
}
4882
}

0 commit comments

Comments
 (0)