From 635b214f861237bcbbb73dc0bbf113c725a7b0de Mon Sep 17 00:00:00 2001 From: Dariusz Stempniak Date: Fri, 27 Oct 2023 14:49:53 +0200 Subject: [PATCH] SNOW-948463 Removed error information for PUT command for GCP (#801) ### Description SNOW-948463 Removed error information for PUT command for GCP ### Checklist - [x] Code compiles correctly - [x] Code is formatted according to [Coding Conventions](../CodingConventions.md) - [x] Created tests which fail without the change (if possible) - [x] All tests passing (`dotnet test`) - [x] Extended the README / documentation, if necessary - [x] Provide JIRA issue id (if possible) or GitHub issue id in PR name --- .../IntegrationTests/SFPutGetTest.cs | 53 +++++++++++++++++-- .../UnitTests/SFAzureClientTest.cs | 4 +- .../UnitTests/SFGCSClientTest.cs | 8 ++- .../FileTransfer/StorageClient/SFGCSClient.cs | 11 ++-- .../StorageClient/SFSnowflakeAzureClient.cs | 4 +- 5 files changed, 66 insertions(+), 14 deletions(-) diff --git a/Snowflake.Data.Tests/IntegrationTests/SFPutGetTest.cs b/Snowflake.Data.Tests/IntegrationTests/SFPutGetTest.cs index 382fabe04..6178f16b8 100644 --- a/Snowflake.Data.Tests/IntegrationTests/SFPutGetTest.cs +++ b/Snowflake.Data.Tests/IntegrationTests/SFPutGetTest.cs @@ -253,6 +253,47 @@ public void TestPutFileRelativePathAsteriskWildcard() } } + [Test] + // presigned url is enabled on CI so we need to disable the test + // it should be enabled when downscoped credential is the default option + [IgnoreOnEnvIs("snowflake_cloud_env", new [] { "GCP" })] + public void TestPutFileWithoutOverwriteFlagSkipsSecondUpload() + { + // Set the PUT query variables + t_inputFilePath = $"{Guid.NewGuid()}.csv"; + t_internalStagePath = $"@{t_schemaName}.{t_stageName}"; + + PrepareFileData(t_inputFilePath); + + using (var conn = new SnowflakeDbConnection(ConnectionString)) + { + conn.Open(); + PutFile(conn, expectedStatus: ResultStatus.UPLOADED); + VerifyFilesAreUploaded(conn, new List { t_inputFilePath }, t_internalStagePath); + PutFile(conn, expectedStatus: ResultStatus.SKIPPED); + } + } + + [Test] + public void TestPutFileWithOverwriteFlagRunsSecondUpload() + { + var overwriteAttribute = "OVERWRITE=TRUE"; + + // Set the PUT query variables + t_inputFilePath = $"{Guid.NewGuid()}.csv"; + t_internalStagePath = $"@{t_schemaName}.{t_stageName}"; + + PrepareFileData(t_inputFilePath); + + using (var conn = new SnowflakeDbConnection(ConnectionString)) + { + conn.Open(); + PutFile(conn, overwriteAttribute, expectedStatus: ResultStatus.UPLOADED); + VerifyFilesAreUploaded(conn, new List { t_inputFilePath }, t_internalStagePath); + PutFile(conn, overwriteAttribute, expectedStatus: ResultStatus.UPLOADED); + } + } + [Test] public void TestPutDirectoryAsteriskWildcard() { @@ -418,13 +459,18 @@ private static bool IsCompressedByTheDriver() } // PUT - upload file from local directory to the stage - private void PutFile(SnowflakeDbConnection conn) + void PutFile( + SnowflakeDbConnection conn, + String additionalAttribute = "", + ResultStatus expectedStatus = ResultStatus.UPLOADED) { using (var command = conn.CreateCommand()) { // Prepare PUT query string putQuery = - $"PUT file://{t_inputFilePath} {t_internalStagePath} AUTO_COMPRESS={(t_autoCompress ? "TRUE" : "FALSE")}"; + $"PUT file://{t_inputFilePath} {t_internalStagePath}" + + $" AUTO_COMPRESS={(t_autoCompress ? "TRUE" : "FALSE")}" + + $" {additionalAttribute}"; // Upload file command.CommandText = putQuery; @@ -432,7 +478,7 @@ private void PutFile(SnowflakeDbConnection conn) Assert.IsTrue(reader.Read()); // Check file status - Assert.AreEqual(ResultStatus.UPLOADED.ToString(), + Assert.AreEqual(expectedStatus.ToString(), reader.GetString((int)SFResultSet.PutGetResponseRowTypeInfo.ResultStatus)); // Check source and destination compression type if (t_autoCompress) @@ -449,6 +495,7 @@ private void PutFile(SnowflakeDbConnection conn) Assert.AreEqual(SFFileCompressionTypes.NONE.Name, reader.GetString((int)SFResultSet.PutGetResponseRowTypeInfo.DestinationCompressionType)); } + Assert.IsNull(reader.GetString((int)SFResultSet.PutGetResponseRowTypeInfo.ErrorDetails)); } } diff --git a/Snowflake.Data.Tests/UnitTests/SFAzureClientTest.cs b/Snowflake.Data.Tests/UnitTests/SFAzureClientTest.cs index 79ef7ed6d..f83351f99 100644 --- a/Snowflake.Data.Tests/UnitTests/SFAzureClientTest.cs +++ b/Snowflake.Data.Tests/UnitTests/SFAzureClientTest.cs @@ -207,7 +207,7 @@ public void TestUploadFile(HttpStatusCode httpStatusCode, ResultStatus expectedR .Returns((blobName) => { var mockBlobClient = new Mock(); - mockBlobClient.Setup(client => client.Upload(It.IsAny())) + mockBlobClient.Setup(client => client.Upload(It.IsAny(), It.IsAny(), It.IsAny())) .Returns(() => MockAzureClient.createMockResponseForBlobContentInfo(key)); return mockBlobClient.Object; @@ -251,7 +251,7 @@ public async Task TestUploadFileAsync(HttpStatusCode httpStatusCode, ResultStatu .Returns((blobName) => { var mockBlobClient = new Mock(); - mockBlobClient.Setup(client => client.UploadAsync(It.IsAny(), It.IsAny())) + mockBlobClient.Setup(client => client.UploadAsync(It.IsAny(), It.IsAny(),It.IsAny())) .Returns(async () => await Task.Run(() => MockAzureClient.createMockResponseForBlobContentInfo(key)).ConfigureAwait(false)); return mockBlobClient.Object; diff --git a/Snowflake.Data.Tests/UnitTests/SFGCSClientTest.cs b/Snowflake.Data.Tests/UnitTests/SFGCSClientTest.cs index f894a82e6..da7c0cdf4 100644 --- a/Snowflake.Data.Tests/UnitTests/SFGCSClientTest.cs +++ b/Snowflake.Data.Tests/UnitTests/SFGCSClientTest.cs @@ -201,12 +201,18 @@ private void AssertForGetFileHeaderTests(ResultStatus expectedResultStatus, File { Assert.AreEqual(MockGCSClient.ContentLength, fileHeader.contentLength); Assert.AreEqual(MockGCSClient.SFCDigest, fileHeader.digest); + Assert.IsNull(_fileMetadata.lastError); + } + else if (expectedResultStatus == ResultStatus.NOT_FOUND_FILE) + { + Assert.IsNull(fileHeader); + Assert.IsNull(_fileMetadata.lastError); } else { Assert.IsNull(fileHeader); + Assert.IsNotNull(_fileMetadata.lastError); } - Assert.AreEqual(expectedResultStatus.ToString(), _fileMetadata.resultStatus); } diff --git a/Snowflake.Data/Core/FileTransfer/StorageClient/SFGCSClient.cs b/Snowflake.Data/Core/FileTransfer/StorageClient/SFGCSClient.cs index 54326f670..14abaad4a 100644 --- a/Snowflake.Data/Core/FileTransfer/StorageClient/SFGCSClient.cs +++ b/Snowflake.Data/Core/FileTransfer/StorageClient/SFGCSClient.cs @@ -444,9 +444,7 @@ private void HandleDownloadResponse(HttpWebResponse response, SFFileMetadata fil private SFFileMetadata HandleFileHeaderErrForPresignedUrls(WebException ex, SFFileMetadata fileMetadata) { Logger.Error("Failed to get file header for presigned url: " + ex.Message); - - fileMetadata.lastError = ex; - + HttpWebResponse response = (HttpWebResponse)ex.Response; if (response.StatusCode == HttpStatusCode.Unauthorized || response.StatusCode == HttpStatusCode.Forbidden || @@ -457,6 +455,7 @@ private SFFileMetadata HandleFileHeaderErrForPresignedUrls(WebException ex, SFFi else { fileMetadata.resultStatus = ResultStatus.ERROR.ToString(); + fileMetadata.lastError = ex; } return fileMetadata; @@ -472,19 +471,18 @@ private SFFileMetadata HandleFileHeaderErrForGeneratedUrls(WebException ex, SFFi { Logger.Error("Failed to get file header for non-presigned url: " + ex.Message); - // If file doesn't exist, GET request fails - fileMetadata.lastError = ex; - HttpWebResponse response = (HttpWebResponse)ex.Response; if (response.StatusCode == HttpStatusCode.Unauthorized) { fileMetadata.resultStatus = ResultStatus.RENEW_TOKEN.ToString(); + fileMetadata.lastError = ex; } else if (response.StatusCode == HttpStatusCode.Forbidden || response.StatusCode == HttpStatusCode.InternalServerError || response.StatusCode == HttpStatusCode.ServiceUnavailable) { fileMetadata.resultStatus = ResultStatus.NEED_RETRY.ToString(); + fileMetadata.lastError = ex; } else if (response.StatusCode == HttpStatusCode.NotFound) { @@ -493,6 +491,7 @@ private SFFileMetadata HandleFileHeaderErrForGeneratedUrls(WebException ex, SFFi else { fileMetadata.resultStatus = ResultStatus.ERROR.ToString(); + fileMetadata.lastError = ex; } return fileMetadata; } diff --git a/Snowflake.Data/Core/FileTransfer/StorageClient/SFSnowflakeAzureClient.cs b/Snowflake.Data/Core/FileTransfer/StorageClient/SFSnowflakeAzureClient.cs index c6834e0df..f0ad3f09e 100644 --- a/Snowflake.Data/Core/FileTransfer/StorageClient/SFSnowflakeAzureClient.cs +++ b/Snowflake.Data/Core/FileTransfer/StorageClient/SFSnowflakeAzureClient.cs @@ -191,7 +191,7 @@ public void UploadFile(SFFileMetadata fileMetadata, Stream fileBytesStream, SFEn { // Issue the POST/PUT request fileBytesStream.Position = 0; - blobClient.Upload(fileBytesStream); + blobClient.Upload(fileBytesStream, overwrite: true); blobClient.SetMetadata(metadata); } catch (RequestFailedException ex) @@ -221,7 +221,7 @@ public async Task UploadFileAsync(SFFileMetadata fileMetadata, Stream fileBytesS { // Issue the POST/PUT request fileBytesStream.Position = 0; - await blobClient.UploadAsync(fileBytesStream, cancellationToken).ConfigureAwait(false); + await blobClient.UploadAsync(fileBytesStream, true, cancellationToken).ConfigureAwait(false); blobClient.SetMetadata(metadata); } catch (RequestFailedException ex)