From b10e21dcfb5c19992cdcedd15e9531fab2db34d4 Mon Sep 17 00:00:00 2001 From: Amanda Nguyen <48961492+amnguye@users.noreply.github.com> Date: Mon, 15 Jul 2024 10:51:56 -0700 Subject: [PATCH] [Storage] Fix for truncating URL paths with "#" (#44941) * WIP - still need to record tests * Recordings * Cleanup * Fixed recordings * Added fix for Shares and Datalake, Updated tests to use OAuth, Updated Changelogs * Rerecord files tests * Merge and record tests --- sdk/storage/Azure.Storage.Blobs/CHANGELOG.md | 1 + sdk/storage/Azure.Storage.Blobs/assets.json | 2 +- .../Azure.Storage.Blobs/src/BlobUriBuilder.cs | 2 +- .../tests/BlobBaseClientTests.cs | 32 +++++++++ .../Azure.Storage.Files.DataLake/CHANGELOG.md | 1 + .../Azure.Storage.Files.DataLake/assets.json | 2 +- .../src/DataLakeUriBuilder.cs | 2 +- .../tests/DirectoryClientTests.cs | 28 ++++++++ .../tests/FileClientTests.cs | 33 +++++++++ .../Azure.Storage.Files.Shares/CHANGELOG.md | 1 + .../Azure.Storage.Files.Shares/assets.json | 2 +- .../src/ShareUriBuilder.cs | 2 +- .../tests/DirectoryClientTests.cs | 27 ++++++++ .../tests/FileClientTests.cs | 68 ++++++++++++++----- 14 files changed, 181 insertions(+), 22 deletions(-) diff --git a/sdk/storage/Azure.Storage.Blobs/CHANGELOG.md b/sdk/storage/Azure.Storage.Blobs/CHANGELOG.md index cee8ee19e8932..16c2f2781fad0 100644 --- a/sdk/storage/Azure.Storage.Blobs/CHANGELOG.md +++ b/sdk/storage/Azure.Storage.Blobs/CHANGELOG.md @@ -7,6 +7,7 @@ ### Breaking Changes ### Bugs Fixed +- Fixed bug where storage clients when constructed with URLs with '#' character would truncate the blob name at the '#'. ### Other Changes diff --git a/sdk/storage/Azure.Storage.Blobs/assets.json b/sdk/storage/Azure.Storage.Blobs/assets.json index 65d82fdcb5979..46a74dc97b014 100644 --- a/sdk/storage/Azure.Storage.Blobs/assets.json +++ b/sdk/storage/Azure.Storage.Blobs/assets.json @@ -2,5 +2,5 @@ "AssetsRepo": "Azure/azure-sdk-assets", "AssetsRepoPrefixPath": "net", "TagPrefix": "net/storage/Azure.Storage.Blobs", - "Tag": "net/storage/Azure.Storage.Blobs_b8e4c4ca1e" + "Tag": "net/storage/Azure.Storage.Blobs_be75b1430c" } diff --git a/sdk/storage/Azure.Storage.Blobs/src/BlobUriBuilder.cs b/sdk/storage/Azure.Storage.Blobs/src/BlobUriBuilder.cs index aa506a6f8c951..1457fe29a13ac 100644 --- a/sdk/storage/Azure.Storage.Blobs/src/BlobUriBuilder.cs +++ b/sdk/storage/Azure.Storage.Blobs/src/BlobUriBuilder.cs @@ -204,7 +204,7 @@ public BlobUriBuilder(Uri uri, bool trimBlobNameSlashes) // Find the account, container, & blob names (if any) if (!string.IsNullOrEmpty(uri.AbsolutePath)) { - var path = uri.GetPath(); + var path = string.Concat(uri.GetPath(), uri.Fragment); var startIndex = 0; diff --git a/sdk/storage/Azure.Storage.Blobs/tests/BlobBaseClientTests.cs b/sdk/storage/Azure.Storage.Blobs/tests/BlobBaseClientTests.cs index 8363d568f7db7..218105b7d38cd 100644 --- a/sdk/storage/Azure.Storage.Blobs/tests/BlobBaseClientTests.cs +++ b/sdk/storage/Azure.Storage.Blobs/tests/BlobBaseClientTests.cs @@ -104,6 +104,38 @@ public async Task Ctor_ConnectionStringEscapeBlobName() Assert.AreEqual(uploadResponse.Value.ETag, propertiesResponse.Value.ETag); } + [RecordedTest] + public async Task Ctor_EscapeBlobName() + { + // Arrange + string blobName = "!*'();[]:@&%=+$,/#äÄöÖüÜß"; + await using DisposingContainer test = await GetTestContainerAsync(); + var data = GetRandomBuffer(Constants.KB); + BlobClient blob = InstrumentClient(test.Container.GetBlobClient(blobName)); + ETag originalEtag; + using (var stream = new MemoryStream(data)) + { + BlobContentInfo info = await blob.UploadAsync(stream); + originalEtag = info.ETag; + } + + // Act + BlobUriBuilder uriBuilder = new BlobUriBuilder(new Uri(Tenants.TestConfigOAuth.BlobServiceEndpoint)) + { + BlobContainerName = blob.BlobContainerName, + BlobName = blobName + }; + BlobBaseClient freshBlobClient = InstrumentClient(new BlobBaseClient( + uriBuilder.ToUri(), + TestEnvironment.Credential, + GetOptions())); + + // Assert + Assert.AreEqual(blobName, freshBlobClient.Name); + BlobProperties propertiesResponse = await freshBlobClient.GetPropertiesAsync(); + Assert.AreEqual(originalEtag, propertiesResponse.ETag); + } + [RecordedTest] public void Ctor_Uri() { diff --git a/sdk/storage/Azure.Storage.Files.DataLake/CHANGELOG.md b/sdk/storage/Azure.Storage.Files.DataLake/CHANGELOG.md index 23694b6f65c04..876725a114639 100644 --- a/sdk/storage/Azure.Storage.Files.DataLake/CHANGELOG.md +++ b/sdk/storage/Azure.Storage.Files.DataLake/CHANGELOG.md @@ -7,6 +7,7 @@ ### Breaking Changes ### Bugs Fixed +- Fixed bug where storage clients when constructed with URLs with '#' character would truncate the path at the '#'. ### Other Changes diff --git a/sdk/storage/Azure.Storage.Files.DataLake/assets.json b/sdk/storage/Azure.Storage.Files.DataLake/assets.json index 597d72370ca4a..442889d04be63 100644 --- a/sdk/storage/Azure.Storage.Files.DataLake/assets.json +++ b/sdk/storage/Azure.Storage.Files.DataLake/assets.json @@ -2,5 +2,5 @@ "AssetsRepo": "Azure/azure-sdk-assets", "AssetsRepoPrefixPath": "net", "TagPrefix": "net/storage/Azure.Storage.Files.DataLake", - "Tag": "net/storage/Azure.Storage.Files.DataLake_133659189e" + "Tag": "net/storage/Azure.Storage.Files.DataLake_186c14971d" } diff --git a/sdk/storage/Azure.Storage.Files.DataLake/src/DataLakeUriBuilder.cs b/sdk/storage/Azure.Storage.Files.DataLake/src/DataLakeUriBuilder.cs index ae789de151eac..8b65f6e888127 100644 --- a/sdk/storage/Azure.Storage.Files.DataLake/src/DataLakeUriBuilder.cs +++ b/sdk/storage/Azure.Storage.Files.DataLake/src/DataLakeUriBuilder.cs @@ -181,7 +181,7 @@ public DataLakeUriBuilder(Uri uri) // Find the share & directory/file path (if any) if (!string.IsNullOrEmpty(uri.AbsolutePath)) { - var path = uri.GetPath(); + var path = string.Concat(uri.GetPath(), uri.Fragment); var startIndex = 0; diff --git a/sdk/storage/Azure.Storage.Files.DataLake/tests/DirectoryClientTests.cs b/sdk/storage/Azure.Storage.Files.DataLake/tests/DirectoryClientTests.cs index 5364051fe5700..74ccc69379729 100644 --- a/sdk/storage/Azure.Storage.Files.DataLake/tests/DirectoryClientTests.cs +++ b/sdk/storage/Azure.Storage.Files.DataLake/tests/DirectoryClientTests.cs @@ -337,6 +337,34 @@ await TestHelper.AssertExpectedExceptionAsync( e => Assert.AreEqual("InvalidAuthenticationInfo", e.ErrorCode)); } + [RecordedTest] + public async Task Ctor_EscapeName() + { + // Arrange + string directoryName = "!*'();[]:@&%=+$,#äÄöÖüÜß"; + await using DisposingFileSystem test = await GetNewFileSystem(); + int size = Constants.KB; + var data = GetRandomBuffer(size); + DataLakeDirectoryClient directory = InstrumentClient(test.Container.GetDirectoryClient(directoryName)); + await directory.CreateAsync(); + + // Act + DataLakeUriBuilder uriBuilder = new DataLakeUriBuilder(new Uri(Tenants.TestConfigHierarchicalNamespace.BlobServiceEndpoint)) + { + FileSystemName = directory.FileSystemName, + DirectoryOrFilePath = directoryName + }; + DataLakeDirectoryClient freshDirectoryClient = InstrumentClient(new DataLakeDirectoryClient( + uriBuilder.ToUri(), + TestEnvironment.Credential, + GetOptions())); + + // Assert + bool exists = await freshDirectoryClient.ExistsAsync(); + Assert.True(exists); + Assert.AreEqual(directoryName, freshDirectoryClient.Name); + } + [RecordedTest] [TestCase(false)] [TestCase(true)] diff --git a/sdk/storage/Azure.Storage.Files.DataLake/tests/FileClientTests.cs b/sdk/storage/Azure.Storage.Files.DataLake/tests/FileClientTests.cs index dade8acee12f6..b08033e1be0e0 100644 --- a/sdk/storage/Azure.Storage.Files.DataLake/tests/FileClientTests.cs +++ b/sdk/storage/Azure.Storage.Files.DataLake/tests/FileClientTests.cs @@ -340,6 +340,39 @@ await TestHelper.AssertExpectedExceptionAsync( e => Assert.AreEqual("InvalidAuthenticationInfo", e.ErrorCode)); } + [RecordedTest] + public async Task Ctor_EscapeName() + { + // Arrange + string fileName = "!*'();[]:@&%=+$,#äÄöÖüÜß"; + await using DisposingFileSystem test = await GetNewFileSystem(); + int size = Constants.KB; + var data = GetRandomBuffer(size); + DataLakeFileClient file = InstrumentClient(test.Container.GetFileClient(fileName)); + ETag originalEtag; + using (var stream = new MemoryStream(data)) + { + PathInfo info = await file.UploadAsync(stream); + originalEtag = info.ETag; + } + + // Act + DataLakeUriBuilder uriBuilder = new DataLakeUriBuilder(new Uri(Tenants.TestConfigHierarchicalNamespace.BlobServiceEndpoint)) + { + FileSystemName = file.FileSystemName, + DirectoryOrFilePath = fileName + }; + DataLakeFileClient freshFileClient = InstrumentClient(new DataLakeFileClient( + uriBuilder.ToUri(), + TestEnvironment.Credential, + GetOptions())); + + // Assert + Assert.AreEqual(fileName, freshFileClient.Name); + PathProperties propertiesResponse = await freshFileClient.GetPropertiesAsync(); + Assert.AreEqual(originalEtag, propertiesResponse.ETag); + } + [RecordedTest] [TestCase(false)] [TestCase(true)] diff --git a/sdk/storage/Azure.Storage.Files.Shares/CHANGELOG.md b/sdk/storage/Azure.Storage.Files.Shares/CHANGELOG.md index 03edc6c7b700f..3c5f19c01376a 100644 --- a/sdk/storage/Azure.Storage.Files.Shares/CHANGELOG.md +++ b/sdk/storage/Azure.Storage.Files.Shares/CHANGELOG.md @@ -7,6 +7,7 @@ ### Breaking Changes ### Bugs Fixed +- Fixed bug where storage clients when constructed with URLs with '#' character would truncate the path at the '#'. ### Other Changes diff --git a/sdk/storage/Azure.Storage.Files.Shares/assets.json b/sdk/storage/Azure.Storage.Files.Shares/assets.json index 028402c5d0c3a..07a2406439a2a 100644 --- a/sdk/storage/Azure.Storage.Files.Shares/assets.json +++ b/sdk/storage/Azure.Storage.Files.Shares/assets.json @@ -2,5 +2,5 @@ "AssetsRepo": "Azure/azure-sdk-assets", "AssetsRepoPrefixPath": "net", "TagPrefix": "net/storage/Azure.Storage.Files.Shares", - "Tag": "net/storage/Azure.Storage.Files.Shares_f7982e093b" + "Tag": "net/storage/Azure.Storage.Files.Shares_093f14c48d" } diff --git a/sdk/storage/Azure.Storage.Files.Shares/src/ShareUriBuilder.cs b/sdk/storage/Azure.Storage.Files.Shares/src/ShareUriBuilder.cs index 38d0830aa5743..99ec7f24621f6 100644 --- a/sdk/storage/Azure.Storage.Files.Shares/src/ShareUriBuilder.cs +++ b/sdk/storage/Azure.Storage.Files.Shares/src/ShareUriBuilder.cs @@ -174,7 +174,7 @@ public ShareUriBuilder(Uri uri) if (!string.IsNullOrEmpty(uri.AbsolutePath)) { // If path starts with a slash, remove it - var path = uri.GetPath(); + var path = string.Concat(uri.GetPath(), uri.Fragment); var startIndex = 0; diff --git a/sdk/storage/Azure.Storage.Files.Shares/tests/DirectoryClientTests.cs b/sdk/storage/Azure.Storage.Files.Shares/tests/DirectoryClientTests.cs index 502e255f4a95c..216204f198cc8 100644 --- a/sdk/storage/Azure.Storage.Files.Shares/tests/DirectoryClientTests.cs +++ b/sdk/storage/Azure.Storage.Files.Shares/tests/DirectoryClientTests.cs @@ -3,6 +3,7 @@ using System; using System.Collections.Generic; +using System.IO; using System.Linq; using System.Threading.Tasks; using Azure.Core.TestFramework; @@ -215,6 +216,32 @@ await TestHelper.AssertExpectedExceptionAsync( e => Assert.AreEqual("InvalidAuthenticationInfo", e.ErrorCode)); } + [RecordedTest] + public async Task Ctor_EscapeDirectoryName() + { + // Arrange + string directoryName = "$=;!#öÖ"; + await using DisposingShare test = await GetTestShareAsync(); + int size = Constants.KB; + var data = GetRandomBuffer(size); + ShareDirectoryClient directory = InstrumentClient(test.Share.GetDirectoryClient(directoryName)); + await directory.CreateAsync(); + + // Act + ShareUriBuilder uriBuilder = new ShareUriBuilder(new Uri(Tenants.TestConfigOAuth.FileServiceEndpoint)) + { + ShareName = directory.ShareName, + DirectoryOrFilePath = directoryName + }; + ShareDirectoryClient freshDirectoryClient = InstrumentClient(new ShareDirectoryClient( + uriBuilder.ToUri(), + TestEnvironment.Credential, + GetOptions())); + + // Assert + Assert.AreEqual(directoryName, freshDirectoryClient.Name); + } + [RecordedTest] public void DirectoryPathsParsing() { diff --git a/sdk/storage/Azure.Storage.Files.Shares/tests/FileClientTests.cs b/sdk/storage/Azure.Storage.Files.Shares/tests/FileClientTests.cs index 434f91a45f602..822df9b47c81c 100644 --- a/sdk/storage/Azure.Storage.Files.Shares/tests/FileClientTests.cs +++ b/sdk/storage/Azure.Storage.Files.Shares/tests/FileClientTests.cs @@ -117,7 +117,7 @@ public async Task Ctor_DefaultAudience() ShareFileClient fileClient = directoryClient.GetFileClient(GetNewFileName()); await fileClient.CreateAsync(Constants.KB); - // Act - Create new blob client with the OAuth Credential and Audience + // Act - Create new Share client with the OAuth Credential and Audience ShareClientOptions options = GetOptionsWithAudience(ShareAudience.DefaultAudience); ShareUriBuilder uriBuilder = new ShareUriBuilder(new Uri(Tenants.TestConfigOAuth.FileServiceEndpoint)) @@ -147,7 +147,7 @@ public async Task Ctor_CustomAudience() ShareFileClient fileClient = directoryClient.GetFileClient(GetNewFileName()); await fileClient.CreateAsync(Constants.KB); - // Act - Create new blob client with the OAuth Credential and Audience + // Act - Create new Share client with the OAuth Credential and Audience ShareClientOptions options = GetOptionsWithAudience(new ShareAudience($"https://{test.Share.AccountName}.file.core.windows.net/")); ShareUriBuilder uriBuilder = new ShareUriBuilder(new Uri(Tenants.TestConfigOAuth.FileServiceEndpoint)) @@ -177,7 +177,7 @@ public async Task Ctor_StorageAccountAudience() ShareFileClient fileClient = directoryClient.GetFileClient(GetNewFileName()); await fileClient.CreateAsync(Constants.KB); - // Act - Create new blob client with the OAuth Credential and Audience + // Act - Create new Share client with the OAuth Credential and Audience ShareClientOptions options = GetOptionsWithAudience(ShareAudience.CreateShareServiceAccountAudience(test.Share.AccountName)); ShareUriBuilder uriBuilder = new ShareUriBuilder(new Uri(Tenants.TestConfigOAuth.FileServiceEndpoint)) @@ -196,6 +196,42 @@ public async Task Ctor_StorageAccountAudience() Assert.IsNotNull(exists); } + [RecordedTest] + public async Task Ctor_EscapeFileName() + { + // Arrange + string fileName = "$=;!#öÖ"; + await using DisposingShare test = await GetTestShareAsync(); + int size = Constants.KB; + var data = GetRandomBuffer(size); + ShareFileClient file = InstrumentClient(test.Share.GetRootDirectoryClient().GetFileClient(fileName)); + ETag originalEtag; + await file.CreateAsync(size); + using (var stream = new MemoryStream(data)) + { + ShareFileUploadInfo info = await file.UploadAsync(stream); + originalEtag = info.ETag; + } + + // Act + ShareUriBuilder uriBuilder = new ShareUriBuilder(new Uri(Tenants.TestConfigOAuth.FileServiceEndpoint)) + { + ShareName = test.Share.Name, + DirectoryOrFilePath = fileName + }; + ShareClientOptions options = GetOptions(); + options.ShareTokenIntent = ShareTokenIntent.Backup; + ShareFileClient freshFileClient = InstrumentClient(new ShareFileClient( + uriBuilder.ToUri(), + TestEnvironment.Credential, + options)); + + // Assert + Assert.AreEqual(fileName, freshFileClient.Name); + ShareFileProperties propertiesResponse = await freshFileClient.GetPropertiesAsync(); + Assert.AreEqual(originalEtag, propertiesResponse.ETag); + } + [RecordedTest] public async Task Ctor_AudienceError() { @@ -206,8 +242,8 @@ public async Task Ctor_AudienceError() ShareFileClient fileClient = directoryClient.GetFileClient(GetNewFileName()); await fileClient.CreateAsync(Constants.KB); - // Act - Create new blob client with the OAuth Credential and Audience - ShareClientOptions options = GetOptionsWithAudience(new ShareAudience("https://badaudience.blob.core.windows.net")); + // Act - Create new Share client with the OAuth Credential and Audience + ShareClientOptions options = GetOptionsWithAudience(new ShareAudience("https://badaudience.Share.core.windows.net")); ShareUriBuilder uriBuilder = new ShareUriBuilder(new Uri(Tenants.TestConfigOAuth.FileServiceEndpoint)) { @@ -4037,7 +4073,7 @@ public async Task ClearRangeAsync_TrailingDot() [TestCase(30 * Constants.KB)] [TestCase(50 * Constants.KB)] [TestCase(501 * Constants.KB)] - public async Task UploadAsync_SmallBlobs(int size) => + public async Task UploadAsync_SmallShares(int size) => // Use a 1KB threshold so we get a lot of individual blocks await UploadAndVerify(size, Constants.KB); @@ -4047,7 +4083,7 @@ public async Task UploadAsync_SmallBlobs(int size) => [TestCase(257 * Constants.MB)] [TestCase(1 * Constants.GB)] [Explicit("https://github.com/Azure/azure-sdk-for-net/issues/9120")] - public async Task UploadAsync_LargeBlobs(int size) => + public async Task UploadAsync_LargeShares(int size) => // TODO: #6781 We don't want to add 1GB of random data in the recordings await UploadAndVerify(size, Constants.MB); @@ -6342,19 +6378,19 @@ public void CanGenerateSas_ClientConstructors() { // Arrange var constants = TestConstants.Create(this); - var blobEndpoint = new Uri("https://127.0.0.1/" + constants.Sas.Account); - var blobSecondaryEndpoint = new Uri("https://127.0.0.1/" + constants.Sas.Account + "-secondary"); - var storageConnectionString = new StorageConnectionString(constants.Sas.SharedKeyCredential, fileStorageUri: (blobEndpoint, blobSecondaryEndpoint)); + var ShareEndpoint = new Uri("https://127.0.0.1/" + constants.Sas.Account); + var ShareSecondaryEndpoint = new Uri("https://127.0.0.1/" + constants.Sas.Account + "-secondary"); + var storageConnectionString = new StorageConnectionString(constants.Sas.SharedKeyCredential, fileStorageUri: (ShareEndpoint, ShareSecondaryEndpoint)); string connectionString = storageConnectionString.ToString(true); - // Act - ShareDirectoryClient(string connectionString, string blobContainerName, string blobName) + // Act - ShareDirectoryClient(string connectionString, string ShareContainerName, string ShareName) ShareFileClient directory = InstrumentClient(new ShareFileClient( connectionString, GetNewShareName(), GetNewDirectoryName())); Assert.IsTrue(directory.CanGenerateSasUri); - // Act - ShareFileClient(string connectionString, string blobContainerName, string blobName, BlobClientOptions options) + // Act - ShareFileClient(string connectionString, string ShareContainerName, string ShareName, ShareClientOptions options) ShareFileClient directory2 = InstrumentClient(new ShareFileClient( connectionString, GetNewShareName(), @@ -6362,15 +6398,15 @@ public void CanGenerateSas_ClientConstructors() GetOptions())); Assert.IsTrue(directory2.CanGenerateSasUri); - // Act - ShareFileClient(Uri blobContainerUri, BlobClientOptions options = default) + // Act - ShareFileClient(Uri ShareContainerUri, ShareClientOptions options = default) ShareFileClient directory3 = InstrumentClient(new ShareFileClient( - blobEndpoint, + ShareEndpoint, GetOptions())); Assert.IsFalse(directory3.CanGenerateSasUri); - // Act - ShareFileClient(Uri blobContainerUri, StorageSharedKeyCredential credential, BlobClientOptions options = default) + // Act - ShareFileClient(Uri ShareContainerUri, StorageSharedKeyCredential credential, ShareClientOptions options = default) ShareFileClient directory4 = InstrumentClient(new ShareFileClient( - blobEndpoint, + ShareEndpoint, constants.Sas.SharedKeyCredential, GetOptions())); Assert.IsTrue(directory4.CanGenerateSasUri);