Skip to content

Commit 05473b8

Browse files
authored
Backport 3.1.x Fix | TDS RPC error on large queries in SqlCommand.ExecuteReaderAsync (#1939)
1 parent e5befdc commit 05473b8

File tree

2 files changed

+141
-3
lines changed

2 files changed

+141
-3
lines changed

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

+3-3
Original file line numberDiff line numberDiff line change
@@ -9860,10 +9860,10 @@ internal Task TdsExecuteRPC(SqlCommand cmd, _SqlRPC[] rpcArray, int timeout, boo
98609860

98619861
// Options
98629862
WriteShort((short)rpcext.options, stateObj);
9863-
}
98649863

9865-
byte[] enclavePackage = cmd.enclavePackage != null ? cmd.enclavePackage.EnclavePackageBytes : null;
9866-
WriteEnclaveInfo(stateObj, enclavePackage);
9864+
byte[] enclavePackage = cmd.enclavePackage != null ? cmd.enclavePackage.EnclavePackageBytes : null;
9865+
WriteEnclaveInfo(stateObj, enclavePackage);
9866+
}
98679867

98689868
// Stream out parameters
98699869
SqlParameter[] parameters = rpcext.parameters;

src/Microsoft.Data.SqlClient/tests/ManualTests/AlwaysEncrypted/ApiShould.cs

+138
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
using System.IO;
99
using System.Linq;
1010
using System.Reflection;
11+
using System.Text;
1112
using System.Threading;
1213
using System.Threading.Tasks;
1314
using Microsoft.Data.SqlClient.AlwaysEncrypted.AzureKeyVaultProvider;
@@ -753,6 +754,55 @@ public void TestExecuteReader(string connection)
753754
});
754755
}
755756

757+
[ConditionalTheory(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringSetupForAE))]
758+
[ClassData(typeof(AEConnectionStringProvider))]
759+
public async void TestExecuteReaderAsyncWithLargeQuery(string connection)
760+
{
761+
string tableName = "VeryLong_01234567890123456789012345678901234567890123456789_TestTableName";
762+
int columnsCount = 50;
763+
764+
// Arrange - drops the table with long name and re-creates it with 52 columns (ID, name, ColumnName0..49)
765+
try
766+
{
767+
DropTableIfExists(connection, tableName);
768+
CreateTable(connection, tableName, columnsCount);
769+
string name = "nobody";
770+
771+
using (SqlConnection sqlConnection = new SqlConnection(connection))
772+
{
773+
await sqlConnection.OpenAsync();
774+
// This creates a "select top 100" query that has over 40k characters
775+
using (SqlCommand sqlCommand = new SqlCommand(GenerateSelectQuery(tableName, columnsCount, 10, "WHERE Name = @FirstName AND ID = @CustomerId"),
776+
sqlConnection,
777+
transaction: null,
778+
columnEncryptionSetting: SqlCommandColumnEncryptionSetting.Enabled))
779+
{
780+
sqlCommand.Parameters.Add(@"CustomerId", SqlDbType.Int);
781+
sqlCommand.Parameters.Add(@"FirstName", SqlDbType.VarChar, name.Length);
782+
783+
sqlCommand.Parameters[0].Value = 0;
784+
sqlCommand.Parameters[1].Value = name;
785+
786+
// Act and Assert
787+
// Test that execute reader async does not throw an exception.
788+
// The table is empty so there should be no results; however, the bug previously found is that it causes a TDS RPC exception on enclave.
789+
using (SqlDataReader sqlDataReader = await sqlCommand.ExecuteReaderAsync())
790+
{
791+
Assert.False(sqlDataReader.HasRows, "The table should be empty");
792+
}
793+
}
794+
}
795+
}
796+
catch (Exception ex)
797+
{
798+
Assert.False(true, $"The following exception was thrown: {ex.Message}");
799+
}
800+
finally
801+
{
802+
DropTableIfExists(connection, tableName);
803+
}
804+
}
805+
756806
[ConditionalTheory(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringSetupForAE))]
757807
[ClassData(typeof(AEConnectionStringProviderWithCommandBehaviorSet1))]
758808
public void TestExecuteReaderWithCommandBehavior(string connection, CommandBehavior commandBehavior)
@@ -2820,6 +2870,94 @@ private void CleanUpTable(string connString, string tableName)
28202870
}
28212871
}
28222872

2873+
2874+
/// <summary>
2875+
/// Creates a table with the specified number of bit columns.
2876+
/// </summary>
2877+
/// <param name="connString">The connection string to the database</param>
2878+
/// <param name="tableName">The table name</param>
2879+
/// <param name="columnsCount">The number of bit columns</param>
2880+
private void CreateTable(string connString, string tableName, int columnsCount)
2881+
{
2882+
using (var sqlConnection = new SqlConnection(connString))
2883+
{
2884+
sqlConnection.Open();
2885+
2886+
SqlCommand cmd = new SqlCommand(GenerateCreateQuery(tableName, columnsCount), sqlConnection);
2887+
cmd.ExecuteNonQuery();
2888+
}
2889+
}
2890+
2891+
/// <summary>
2892+
/// Drops the table if the specified table exists
2893+
/// </summary>
2894+
/// <param name="connString">The connection string to the database</param>
2895+
/// <param name="tableName">The name of the table to be dropped</param>
2896+
private void DropTableIfExists(string connString, string tableName)
2897+
{
2898+
using (var sqlConnection = new SqlConnection(connString))
2899+
{
2900+
sqlConnection.Open();
2901+
SqlCommand cmd = new SqlCommand($"DROP TABLE IF EXISTS {tableName};", sqlConnection);
2902+
cmd.ExecuteNonQuery();
2903+
}
2904+
}
2905+
2906+
/// <summary>
2907+
/// Generates the query for creating a table with the number of bit columns specified.
2908+
/// </summary>
2909+
/// <param name="tableName">The name of the table</param>
2910+
/// <param name="columnsCount">The number of columns for the table</param>
2911+
/// <returns></returns>
2912+
private string GenerateCreateQuery(string tableName, int columnsCount)
2913+
{
2914+
StringBuilder builder = new StringBuilder();
2915+
builder.Append(string.Format("CREATE TABLE [dbo].[{0}]", tableName));
2916+
builder.Append('(');
2917+
builder.AppendLine("[ID][bigint] NOT NULL,");
2918+
builder.AppendLine("[Name] [varchar] (200) NOT NULL");
2919+
for (int i = 0; i < columnsCount; i++)
2920+
{
2921+
builder.Append(',');
2922+
builder.Append($"[ColumnName{i}][bit] NULL");
2923+
}
2924+
builder.Append(");");
2925+
2926+
return builder.ToString();
2927+
}
2928+
2929+
/// <summary>
2930+
/// Generates the large query with the select top 100 of all the columns repeated multiple times.
2931+
/// </summary>
2932+
/// <param name="tableName">The name of the table</param>
2933+
/// <param name="columnsCount">The number of columns to be explicitly included</param>
2934+
/// <param name="repeat">The number of times the select query is repeated</param>
2935+
/// <param name="where">A where clause for additional filters</param>
2936+
/// <returns></returns>
2937+
private string GenerateSelectQuery(string tableName, int columnsCount, int repeat = 10, string where = "")
2938+
{
2939+
StringBuilder builder = new StringBuilder();
2940+
builder.AppendLine($"SELECT TOP 100");
2941+
builder.AppendLine($"[{tableName}].[ID],");
2942+
builder.AppendLine($"[{tableName}].[Name]");
2943+
for (int i = 0; i < columnsCount; i++)
2944+
{
2945+
builder.Append(",");
2946+
builder.AppendLine($"[{tableName}].[ColumnName{i}]");
2947+
}
2948+
2949+
string extra = string.IsNullOrEmpty(where) ? $"(NOLOCK) [{tableName}]" : where;
2950+
builder.AppendLine($"FROM [{tableName}] {extra};");
2951+
2952+
StringBuilder builder2 = new StringBuilder();
2953+
for (int i = 0; i < repeat; i++)
2954+
{
2955+
builder2.AppendLine(builder.ToString());
2956+
}
2957+
2958+
return builder2.ToString();
2959+
}
2960+
28232961
/// <summary>
28242962
/// An helper method to test the cancellation of the command using cancellationToken to async SqlCommand APIs.
28252963
/// </summary>

0 commit comments

Comments
 (0)