|
8 | 8 | using System.IO;
|
9 | 9 | using System.Linq;
|
10 | 10 | using System.Reflection;
|
| 11 | +using System.Text; |
11 | 12 | using System.Threading;
|
12 | 13 | using System.Threading.Tasks;
|
13 | 14 | using Microsoft.Data.SqlClient.AlwaysEncrypted.AzureKeyVaultProvider;
|
@@ -753,6 +754,55 @@ public void TestExecuteReader(string connection)
|
753 | 754 | });
|
754 | 755 | }
|
755 | 756 |
|
| 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 | + |
756 | 806 | [ConditionalTheory(typeof(DataTestUtility), nameof(DataTestUtility.AreConnStringSetupForAE))]
|
757 | 807 | [ClassData(typeof(AEConnectionStringProviderWithCommandBehaviorSet1))]
|
758 | 808 | public void TestExecuteReaderWithCommandBehavior(string connection, CommandBehavior commandBehavior)
|
@@ -2820,6 +2870,94 @@ private void CleanUpTable(string connString, string tableName)
|
2820 | 2870 | }
|
2821 | 2871 | }
|
2822 | 2872 |
|
| 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 | + |
2823 | 2961 | /// <summary>
|
2824 | 2962 | /// An helper method to test the cancellation of the command using cancellationToken to async SqlCommand APIs.
|
2825 | 2963 | /// </summary>
|
|
0 commit comments