From 8d82464e74bae7ac0a3da236becac61320165299 Mon Sep 17 00:00:00 2001 From: Christopher Scott Date: Tue, 12 Nov 2024 15:20:54 -0600 Subject: [PATCH 1/7] wip --- .../tests/TableClientQueryExpressionTests.cs | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs b/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs index 6582a0043319c..66747ec55673d 100644 --- a/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs +++ b/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs @@ -23,6 +23,7 @@ public class TableClientQueryExpressionTests private const string TableName = "someTableName"; private const string TableName2 = "otherTableName"; private const string Partition = "partition"; + private const string MixedCasePK = "PartitionKey"; private const string Row = "row"; private const int SomeInt = 10; private const double SomeDouble = 10.10; @@ -94,6 +95,8 @@ public class TableClientQueryExpressionTests private static readonly Expression> s_tableEntExpImplicitBooleanCmpCasted = ent => (bool)ent.GetBoolean("Bool"); private static readonly Expression> s_tableEntExpImplicitBooleanCmpOr = ent => ent.GetBoolean("Bool").Value || (bool)ent.GetBoolean("Bool"); private static readonly Expression> s_tableEntExpImplicitBooleanCmpNot = ent => !ent.GetBoolean("Bool").Value; + private static readonly Expression> s_tableEntExpEquals = ent => ent.PartitionKey.Equals(MixedCasePK); + private static readonly Expression> s_tableEntExpEqualsOIC = ent => ent.PartitionKey.Equals(MixedCasePK, StringComparison.OrdinalIgnoreCase); public static object[] TableEntityExpressionTestCases = { @@ -125,7 +128,7 @@ public class TableClientQueryExpressionTests new object[] { $"not (Bool eq true)", s_complexExpImplicitBooleanCmpNot }, new object[] { $"BoolN eq true", s_complexExpImplicitCastedNullableBooleanCmp }, new object[] { $"(Bool eq true) and (BoolN eq true)", s_complexExpImplicitBooleanCmpAnd }, - new object[] { $"(Bool eq true) or (BoolN eq true)", s_complexExpImplicitCastedBooleanCmpOr } + new object[] { $"(Bool eq true) or (BoolN eq true)", s_complexExpImplicitCastedBooleanCmpOr }, }; public static object[] TableItemExpressionTestCases = @@ -162,7 +165,9 @@ public class TableClientQueryExpressionTests new object[] { $"Bool eq true", s_tableEntExpImplicitBooleanCmp }, new object[] { $"Bool eq true", s_tableEntExpImplicitBooleanCmpCasted }, new object[] { $"(Bool eq true) or (Bool eq true)", s_tableEntExpImplicitBooleanCmpOr }, - new object[] { $"not (Bool eq true)", s_tableEntExpImplicitBooleanCmpNot } + new object[] { $"not (Bool eq true)", s_tableEntExpImplicitBooleanCmpNot }, + new object[] { $"PartitionKey eq '{MixedCasePK}'", s_tableEntExpEquals }, + new object[] { $"PartitionKey eq '{MixedCasePK.ToLower()}'", s_tableEntExpEqualsOIC }, }; [TestCaseSource(nameof(TableItemExpressionTestCases))] From ee7687cb3b2293373c12982812c69344b4a9c762 Mon Sep 17 00:00:00 2001 From: Christopher Scott Date: Thu, 14 Nov 2024 16:51:20 -0600 Subject: [PATCH 2/7] fix provisioning template --- sdk/tables/test-resources.json | 66 +++++++++++++++++++++++++++------- 1 file changed, 54 insertions(+), 12 deletions(-) diff --git a/sdk/tables/test-resources.json b/sdk/tables/test-resources.json index 7d77065dbf796..bf462a2b9e968 100644 --- a/sdk/tables/test-resources.json +++ b/sdk/tables/test-resources.json @@ -1,5 +1,5 @@ { - "$schema": "https://schema.management.azure.com/schemas/2015-01-01/deploymentTemplate.json#", + "$schema": "https://schema.management.azure.com/schemas/2019-04-01/deploymentTemplate.json#", "contentVersion": "1.0.0.0", "parameters": { "baseName": { @@ -12,18 +12,18 @@ } }, "cosmosEndpointSuffix": { - "type": "string", - "defaultValue": "cosmos.azure.com", - "metadata": { - "description": "The url suffix to use when accessing the cosmos data plane." - } + "type": "string", + "defaultValue": "cosmos.azure.com", + "metadata": { + "description": "The url suffix to use when accessing the cosmos data plane." + } }, "storageEndpointSuffix": { - "type": "string", - "defaultValue": "core.windows.net", - "metadata": { - "description": "The url suffix to use when accessing the storage data plane." - } + "type": "string", + "defaultValue": "core.windows.net", + "metadata": { + "description": "The url suffix to use when accessing the storage data plane." + } } }, "variables": { @@ -48,7 +48,16 @@ "virtualNetworkRules": [], "ipRules": [], "defaultAction": "Allow" - } + }, + "customCosmosRoleName": "Azure Cosmos DB SDK role for Table Data Plane", + "customCosmosRoleDescription": "Azure Cosmos DB SDK role for Table Data Plane", + "customCosmosRoleActions": [ + "Microsoft.DocumentDB/databaseAccounts/readMetadata", + "Microsoft.DocumentDB/databaseAccounts/tables/*", + "Microsoft.DocumentDB/databaseAccounts/tables/containers/*", + "Microsoft.DocumentDB/databaseAccounts/tables/containers/entities/*", + "Microsoft.DocumentDB/databaseAccounts/throughputSettings/read" + ] }, "resources": [ { @@ -118,6 +127,39 @@ ], "ipRules": [] } + }, + { + "dependsOn": [ + "[resourceId('Microsoft.DocumentDB/databaseAccounts', variables('primaryAccountName'))]" + ], + "type": "Microsoft.DocumentDB/databaseAccounts/tableRoleDefinitions", + "apiVersion": "2024-05-15", + "name": "[concat(variables('primaryAccountName'), '/', guid(variables('customCosmosRoleName')))]", + "properties": { + "roleName": "[variables('customCosmosRoleName')]", + "description": "[variables('customCosmosRoleDescription')]", + "permissions": [ + { + "dataActions": "[variables('customCosmosRoleActions')]" + } + ], + "assignableScopes": [ + "[concat('/subscriptions/', subscription().subscriptionId, '/resourceGroups/', resourceGroup().name, '/providers/Microsoft.DocumentDB/databaseAccounts/', variables('primaryAccountName'))]" + ] + } + }, + { + "dependsOn": [ + "[resourceId('Microsoft.DocumentDB/databaseAccounts/tableRoleDefinitions', variables('primaryAccountName'), guid(variables('customCosmosRoleName')))]" + ], + "type": "Microsoft.DocumentDB/databaseAccounts/tableRoleAssignments", + "apiVersion": "2024-05-15", + "name": "[concat(variables('primaryAccountName'), '/', guid(variables('customCosmosRoleName')))]", + "properties": { + "scope": "[concat('/subscriptions/', subscription().subscriptionId, '/resourceGroups/', resourceGroup().name, '/providers/Microsoft.DocumentDB/databaseAccounts/', variables('primaryAccountName'))]", + "roleDefinitionId": "[resourceId('Microsoft.DocumentDB/databaseAccounts/tableRoleDefinitions', variables('primaryAccountName'), guid(variables('customCosmosRoleName')))]", + "principalId": "[parameters('testApplicationOid')]" + } } ], "outputs": { From e879e140cee2a0ac69e9292a7ec95edb103090a7 Mon Sep 17 00:00:00 2001 From: Christopher Scott Date: Thu, 14 Nov 2024 18:02:39 -0600 Subject: [PATCH 3/7] Throw on query filters with unsupported method overloads --- sdk/tables/Azure.Data.Tables/CHANGELOG.md | 2 ++ .../src/Queryable/ExpressionNormalizer.cs | 12 +++++++ .../Azure.Data.Tables/src/TableClient.cs | 31 ++++++++++++++++++- .../tests/TableClientQueryExpressionTests.cs | 31 ++++++++++++++++--- 4 files changed, 70 insertions(+), 6 deletions(-) diff --git a/sdk/tables/Azure.Data.Tables/CHANGELOG.md b/sdk/tables/Azure.Data.Tables/CHANGELOG.md index 126e935a25663..08e33625c427c 100644 --- a/sdk/tables/Azure.Data.Tables/CHANGELOG.md +++ b/sdk/tables/Azure.Data.Tables/CHANGELOG.md @@ -6,6 +6,8 @@ ### Breaking Changes +- Calling `TableClient.Query`, `TableClient.QueryAsync`, or `TableClient.CreateQueryFilter` with a filter expression that uses `string.Equals` or `string.Compare` with a `StringComparison` parameter will now throw an exception. This is because the Azure Table service does not support these methods in query filters. Previously the `StringComparison` argument was silently ignored, which can lead to subtle bugs in client code. + ### Bugs Fixed ### Other Changes diff --git a/sdk/tables/Azure.Data.Tables/src/Queryable/ExpressionNormalizer.cs b/sdk/tables/Azure.Data.Tables/src/Queryable/ExpressionNormalizer.cs index 0e0bdcc98fe3a..1bfe6237cd08b 100644 --- a/sdk/tables/Azure.Data.Tables/src/Queryable/ExpressionNormalizer.cs +++ b/sdk/tables/Azure.Data.Tables/src/Queryable/ExpressionNormalizer.cs @@ -162,11 +162,19 @@ internal Expression VisitMethodCallNoRewrite(MethodCallExpression call) if (visited.Method.IsStatic && visited.Method.Name == "Equals" && visited.Arguments.Count > 1) { + if (visited.Arguments.Count > 2) + { + throw new NotSupportedException("string.Equals method with more than two arguments is not supported."); + } return Expression.Equal(visited.Arguments[0], visited.Arguments[1], false, visited.Method); } if (!visited.Method.IsStatic && visited.Method.Name == "Equals" && visited.Arguments.Count > 0) { + if (visited.Arguments.Count > 1) + { + throw new NotSupportedException("Equals method with more than two arguments is not supported."); + } return CreateRelationalOperator(ExpressionType.Equal, visited.Object, visited.Arguments[0]); } @@ -182,6 +190,10 @@ internal Expression VisitMethodCallNoRewrite(MethodCallExpression call) if (visited.Method.IsStatic && visited.Method.Name == "Compare" && visited.Arguments.Count > 1 && visited.Method.ReturnType == typeof(int)) { + if (visited.Arguments.Count > 2) + { + throw new NotSupportedException("string.Compare method with more than two arguments is not supported."); + } return CreateCompareExpression(visited.Arguments[0], visited.Arguments[1]); } diff --git a/sdk/tables/Azure.Data.Tables/src/TableClient.cs b/sdk/tables/Azure.Data.Tables/src/TableClient.cs index 9bc1ed3defed0..d89797aa15a12 100644 --- a/sdk/tables/Azure.Data.Tables/src/TableClient.cs +++ b/sdk/tables/Azure.Data.Tables/src/TableClient.cs @@ -1041,6 +1041,15 @@ public virtual Response UpdateEntity(T entity, ETag ifMatch, TableUpdateMode /// /// Returns only entities that satisfy the specified filter expression. /// For example, the following expression would filter entities with a PartitionKey of 'foo': e => e.PartitionKey == "foo". + /// + /// The following string comparison methods are supported as part of a filter expression: + /// + /// + /// + /// + /// + /// + /// /// /// /// The maximum number of entities that will be returned per page. If unspecified, the default value is 1000 for storage accounts and is not limited for Cosmos DB table API. @@ -1079,6 +1088,15 @@ public virtual AsyncPageable QueryAsync( /// /// Returns only entities that satisfy the specified filter expression. /// For example, the following expression would filter entities with a PartitionKey of 'foo': e => e.PartitionKey == "foo". + /// + /// The following string comparison methods are supported as part of a filter expression: + /// + /// + /// + /// + /// + /// + /// /// /// /// The maximum number of entities that will be returned per page. If unspecified, the default value is 1000 for storage accounts and is not limited for Cosmos DB table API. @@ -1494,7 +1512,18 @@ public virtual Response SetAccessPolicy(IEnumerable table /// /// The type of the entity being queried. Typically this will be derived from /// or . - /// A filter expression. + /// A filter expression. + /// for example: e => e.PartitionKey == "foo". + /// + /// The following string comparison methods are supported as part of a filter expression: + /// + /// + /// + /// + /// + /// + /// + /// /// The string representation of the filter expression. public static string CreateQueryFilter(Expression> filter) => Bind(filter); diff --git a/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs b/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs index 66747ec55673d..d3799dc0571ee 100644 --- a/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs +++ b/sdk/tables/Azure.Data.Tables/tests/TableClientQueryExpressionTests.cs @@ -95,8 +95,13 @@ public class TableClientQueryExpressionTests private static readonly Expression> s_tableEntExpImplicitBooleanCmpCasted = ent => (bool)ent.GetBoolean("Bool"); private static readonly Expression> s_tableEntExpImplicitBooleanCmpOr = ent => ent.GetBoolean("Bool").Value || (bool)ent.GetBoolean("Bool"); private static readonly Expression> s_tableEntExpImplicitBooleanCmpNot = ent => !ent.GetBoolean("Bool").Value; - private static readonly Expression> s_tableEntExpEquals = ent => ent.PartitionKey.Equals(MixedCasePK); - private static readonly Expression> s_tableEntExpEqualsOIC = ent => ent.PartitionKey.Equals(MixedCasePK, StringComparison.OrdinalIgnoreCase); + private static readonly Expression> s_tableEntExpEquals = ent => ent.PartitionKey.Equals(Partition); + private static readonly Expression> s_CEtableEntExpEquals = ent => ent.String.Equals(Partition); + private static readonly Expression> s_CEtableEntExpStaticEquals = ent => string.Equals(ent.String, Partition); + private static readonly Expression> s_TEequalsUnsupported = ent => ent.PartitionKey.Equals(Partition, StringComparison.OrdinalIgnoreCase); + private static readonly Expression> s_TEequalsToLowerUnsupported = ent => ent.PartitionKey.Equals(Partition.ToLower(), StringComparison.OrdinalIgnoreCase); + private static readonly Expression> s_TEequalsStaticUnsupported = ent => string.Equals(ent.PartitionKey, Partition, StringComparison.OrdinalIgnoreCase); + private static readonly Expression> s_TECompareStaticUnsupported = ent => string.Compare(ent.PartitionKey, Partition, StringComparison.OrdinalIgnoreCase) > 0; public static object[] TableEntityExpressionTestCases = { @@ -128,7 +133,9 @@ public class TableClientQueryExpressionTests new object[] { $"not (Bool eq true)", s_complexExpImplicitBooleanCmpNot }, new object[] { $"BoolN eq true", s_complexExpImplicitCastedNullableBooleanCmp }, new object[] { $"(Bool eq true) and (BoolN eq true)", s_complexExpImplicitBooleanCmpAnd }, - new object[] { $"(Bool eq true) or (BoolN eq true)", s_complexExpImplicitCastedBooleanCmpOr }, + new object[] { $"(Bool eq true) or (BoolN eq true)", s_complexExpImplicitCastedBooleanCmpOr }, + new object[] { $"String eq '{Partition}'", s_CEtableEntExpEquals }, + new object[] { $"String eq '{Partition}'", s_CEtableEntExpStaticEquals }, }; public static object[] TableItemExpressionTestCases = @@ -166,8 +173,15 @@ public class TableClientQueryExpressionTests new object[] { $"Bool eq true", s_tableEntExpImplicitBooleanCmpCasted }, new object[] { $"(Bool eq true) or (Bool eq true)", s_tableEntExpImplicitBooleanCmpOr }, new object[] { $"not (Bool eq true)", s_tableEntExpImplicitBooleanCmpNot }, - new object[] { $"PartitionKey eq '{MixedCasePK}'", s_tableEntExpEquals }, - new object[] { $"PartitionKey eq '{MixedCasePK.ToLower()}'", s_tableEntExpEqualsOIC }, + new object[] { $"PartitionKey eq '{Partition}'", s_tableEntExpEquals }, + }; + + public static object[] UnSupportedTableItemExpressionTestCases = + { + new object[] { s_TEequalsUnsupported }, + new object[] { s_TEequalsStaticUnsupported }, + new object[] { s_TEequalsToLowerUnsupported }, + new object[] { s_TECompareStaticUnsupported }, }; [TestCaseSource(nameof(TableItemExpressionTestCases))] @@ -196,5 +210,12 @@ public void TestDictionaryTableEntityFilterExpressions(string expectedFilter, Ex Assert.That(filter, Is.EqualTo(expectedFilter)); } + + [TestCaseSource(nameof(UnSupportedTableItemExpressionTestCases))] + [Test] + public void TestTableItemFilterExpressionsUnsupported(Expression> expression) + { + Assert.Throws(() => TableClient.CreateQueryFilter(expression)); + } } } From 61fc3a97e1a33ce53d8eac0e0d6ad39bddaeae37 Mon Sep 17 00:00:00 2001 From: Christopher Scott Date: Fri, 15 Nov 2024 11:42:30 -0600 Subject: [PATCH 4/7] fix test --- sdk/tables/Azure.Data.Tables/tests/samples/Sample0_Auth.cs | 5 +---- .../tests/samples/Sample8_CustomizingSerialization.cs | 5 +---- 2 files changed, 2 insertions(+), 8 deletions(-) diff --git a/sdk/tables/Azure.Data.Tables/tests/samples/Sample0_Auth.cs b/sdk/tables/Azure.Data.Tables/tests/samples/Sample0_Auth.cs index a6b3711a95c75..31b9360f51785 100644 --- a/sdk/tables/Azure.Data.Tables/tests/samples/Sample0_Auth.cs +++ b/sdk/tables/Azure.Data.Tables/tests/samples/Sample0_Auth.cs @@ -108,10 +108,7 @@ public async Task TokenCredentialAuth() #if SNIPPET new DefaultAzureCredential()); #else - new ClientSecretCredential( - GetVariable("TENANT_ID"), - GetVariable("CLIENT_ID"), - GetVariable("CLIENT_SECRET"))); + Credential); #endif // Create the table if it doesn't already exist to verify we've successfully authenticated. diff --git a/sdk/tables/Azure.Data.Tables/tests/samples/Sample8_CustomizingSerialization.cs b/sdk/tables/Azure.Data.Tables/tests/samples/Sample8_CustomizingSerialization.cs index bc7972473ac9d..01a9c9a350c38 100644 --- a/sdk/tables/Azure.Data.Tables/tests/samples/Sample8_CustomizingSerialization.cs +++ b/sdk/tables/Azure.Data.Tables/tests/samples/Sample8_CustomizingSerialization.cs @@ -28,10 +28,7 @@ public async Task CustomizeSerialization() #if SNIPPET new DefaultAzureCredential()); #else - new ClientSecretCredential( - GetVariable("TENANT_ID"), - GetVariable("CLIENT_ID"), - GetVariable("CLIENT_SECRET"))); + Credential); #endif // Create the table if it doesn't already exist. From 101002258103360eaa370138a8574728b768bab7 Mon Sep 17 00:00:00 2001 From: Christopher Scott Date: Fri, 15 Nov 2024 11:49:12 -0600 Subject: [PATCH 5/7] fix docs --- .../Azure.Data.Tables/src/TableClient.cs | 24 +++++++++---------- 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/sdk/tables/Azure.Data.Tables/src/TableClient.cs b/sdk/tables/Azure.Data.Tables/src/TableClient.cs index d89797aa15a12..8b2346e1727c1 100644 --- a/sdk/tables/Azure.Data.Tables/src/TableClient.cs +++ b/sdk/tables/Azure.Data.Tables/src/TableClient.cs @@ -1044,10 +1044,10 @@ public virtual Response UpdateEntity(T entity, ETag ifMatch, TableUpdateMode /// /// The following string comparison methods are supported as part of a filter expression: /// - /// - /// - /// - /// + /// + /// + /// + /// /// /// /// @@ -1091,10 +1091,10 @@ public virtual AsyncPageable QueryAsync( /// /// The following string comparison methods are supported as part of a filter expression: /// - /// - /// - /// - /// + /// + /// + /// + /// /// /// /// @@ -1517,10 +1517,10 @@ public virtual Response SetAccessPolicy(IEnumerable table /// /// The following string comparison methods are supported as part of a filter expression: /// - /// - /// - /// - /// + /// + /// + /// + /// /// /// /// From db4612ac86e3a68edea7763aed01314cf369a8fc Mon Sep 17 00:00:00 2001 From: Christopher Scott Date: Fri, 15 Nov 2024 14:07:39 -0600 Subject: [PATCH 6/7] handle cosmos connection limit errors in live tests --- .../Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs b/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs index 72c43e1c9f11a..f8bddc4a26d0b 100644 --- a/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs +++ b/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs @@ -340,6 +340,10 @@ protected async Task CosmosThrottleWrapper(Func> delay *= 2; } } + catch (RequestFailedException ex) when (ex.Status == 401 && ex.GetRawResponse().Content.ToString().Contains("Exceeded allowed aad token limit")) + { + Assert.Ignore("Exceeded allowed aad token limit of Cosmos table API. See https://github.com/Azure/azure-sdk-for-net/issues/47191 for details."); + } } } From 0836fe200ca7435c66202548239ad9a341eb7c07 Mon Sep 17 00:00:00 2001 From: Christopher Scott Date: Fri, 15 Nov 2024 15:37:48 -0600 Subject: [PATCH 7/7] remove cosmos handling.. need another approach --- .../Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs | 4 ---- 1 file changed, 4 deletions(-) diff --git a/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs b/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs index f8bddc4a26d0b..72c43e1c9f11a 100644 --- a/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs +++ b/sdk/tables/Azure.Data.Tables/tests/TableServiceLiveTestsBase.cs @@ -340,10 +340,6 @@ protected async Task CosmosThrottleWrapper(Func> delay *= 2; } } - catch (RequestFailedException ex) when (ex.Status == 401 && ex.GetRawResponse().Content.ToString().Contains("Exceeded allowed aad token limit")) - { - Assert.Ignore("Exceeded allowed aad token limit of Cosmos table API. See https://github.com/Azure/azure-sdk-for-net/issues/47191 for details."); - } } }