Skip to content

Commit

Permalink
Merge branch 'master' into SNOW-715504-MFA-Token-Cache
Browse files Browse the repository at this point in the history
  • Loading branch information
sfc-gh-knozderko committed Dec 9, 2024
2 parents aba9cf4 + a33bb45 commit 26f52ee
Show file tree
Hide file tree
Showing 18 changed files with 440 additions and 62 deletions.
9 changes: 6 additions & 3 deletions .github/workflows/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ jobs:
strategy:
fail-fast: false
matrix:
dotnet: ['net6.0', 'net7.0', 'net8.0', 'net462', 'net471', 'net472', 'net48', 'net481']
dotnet: ['net6.0', 'net7.0', 'net8.0', 'net9.0', 'net462', 'net471', 'net472', 'net48', 'net481']
cloud_env: ['AZURE', 'GCP', 'AWS']
steps:
- name: Checkout code
Expand All @@ -42,6 +42,7 @@ jobs:
dotnet-version: |
6.0.x
8.0.x
9.0.x
dotnet-quality: 'ga'
- name: Setup Python
uses: actions/setup-python@v5
Expand Down Expand Up @@ -93,7 +94,7 @@ jobs:
strategy:
fail-fast: false
matrix:
dotnet: ['net6.0', 'net7.0', 'net8.0']
dotnet: ['net6.0', 'net7.0', 'net8.0', 'net9.0']
cloud_env: ['AZURE', 'GCP', 'AWS']
steps:
- uses: actions/checkout@v4
Expand All @@ -103,6 +104,7 @@ jobs:
dotnet-version: |
6.0.x
8.0.x
9.0.x
dotnet-quality: 'ga'
- name: Setup Python
uses: actions/setup-python@v5
Expand Down Expand Up @@ -152,7 +154,7 @@ jobs:
strategy:
fail-fast: false
matrix:
dotnet: ['net6.0', 'net7.0', 'net8.0']
dotnet: ['net6.0', 'net7.0', 'net8.0', 'net9.0']
cloud_env: ['AZURE', 'GCP', 'AWS']
steps:
- uses: actions/checkout@v4
Expand All @@ -162,6 +164,7 @@ jobs:
dotnet-version: |
6.0.x
8.0.x
9.0.x
dotnet-quality: 'ga'
- name: Setup Python
uses: actions/setup-python@v5
Expand Down
2 changes: 1 addition & 1 deletion Jenkinsfile
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ timestamps {
string(name: 'branch', value: 'main'),
string(name: 'client_git_commit', value: scmInfo.GIT_COMMIT),
string(name: 'client_git_branch', value: scmInfo.GIT_BRANCH),
string(name: 'TARGET_DOCKER_TEST_IMAGE', value: 'dotnet-ubuntu204-net8'),
string(name: 'TARGET_DOCKER_TEST_IMAGE', value: 'dotnet-ubuntu204-net9'),
string(name: 'parent_job', value: env.JOB_NAME),
string(name: 'parent_build_number', value: env.BUILD_NUMBER)
]
Expand Down
3 changes: 2 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
[![NuGet](https://img.shields.io/nuget/v/Snowflake.Data.svg)](https://www.nuget.org/packages/Snowflake.Data/)
[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0)

The Snowflake .NET connector supports the the following .NET framework and libraries versions:
The Snowflake .NET connector supports the following .NET framework and libraries versions:

- .NET Framework 4.6.2
- .NET Framework 4.7.1
Expand All @@ -14,6 +14,7 @@ The Snowflake .NET connector supports the the following .NET framework and libra
- .NET 6.0
- .NET 7.0
- .NET 8.0
- .NET 9.0

Disclaimer: While the connector targets netstandard2.0 and may work with versions in its [support matrix](https://learn.microsoft.com/en-us/dotnet/standard/net-standard?tabs=net-standard-2-0#select-net-standard-version), only the versions listed above are supported and tested by the connector

Expand Down
4 changes: 2 additions & 2 deletions Snowflake.Data.Tests/Snowflake.Data.Tests.csproj
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFrameworks>net6.0;net7.0;net8.0;net462;net471;net472;net48;net481</TargetFrameworks>
<TargetFrameworks Condition="'$(OS)' != 'Windows_NT'">net6.0;net7.0;net8.0;</TargetFrameworks>
<TargetFrameworks>net6.0;net7.0;net8.0;net9.0;net462;net471;net472;net48;net481</TargetFrameworks>
<TargetFrameworks Condition="'$(OS)' != 'Windows_NT'">net6.0;net7.0;net8.0;net9.0</TargetFrameworks>
<Title>Snowflake.Data.Tests</Title>
<Description>Snowflake Connector for .NET</Description>
<Company>Snowflake Computing, Inc</Company>
Expand Down
63 changes: 63 additions & 0 deletions Snowflake.Data.Tests/UnitTests/PutGetStageInfoTest.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
/*
* Copyright (c) 2024 Snowflake Computing Inc. All rights reserved.
*/

using System.Collections.Generic;
using NUnit.Framework;
using Snowflake.Data.Core;
using Snowflake.Data.Core.FileTransfer;

namespace Snowflake.Data.Tests.UnitTests
{
[TestFixture]
public class PutGetStageInfoTest
{
[Test]
[TestCaseSource(nameof(TestCases))]
public void TestGcsRegionalUrl(string region, bool useRegionalUrl, string endPoint, string expectedGcsEndpoint)
{
// arrange
var stageInfo = CreateGcsStageInfo(region, useRegionalUrl, endPoint);

// act
var gcsCustomEndpoint = stageInfo.GcsCustomEndpoint();

// assert
Assert.AreEqual(expectedGcsEndpoint, gcsCustomEndpoint);
}

internal static IEnumerable<object[]> TestCases()
{
yield return new object[] { "US-CENTRAL1", false, null, null };
yield return new object[] { "US-CENTRAL1", false, "", null };
yield return new object[] { "US-CENTRAL1", false, "null", null };
yield return new object[] { "US-CENTRAL1", false, " ", null };
yield return new object[] { "US-CENTRAL1", false, "example.com", "example.com" };
yield return new object[] { "ME-CENTRAL2", false, null, "storage.me-central2.rep.googleapis.com" };
yield return new object[] { "ME-CENTRAL2", true, null, "storage.me-central2.rep.googleapis.com" };
yield return new object[] { "ME-CENTRAL2", true, "", "storage.me-central2.rep.googleapis.com" };
yield return new object[] { "ME-CENTRAL2", true, " ", "storage.me-central2.rep.googleapis.com" };
yield return new object[] { "ME-CENTRAL2", true, "example.com", "example.com" };
yield return new object[] { "US-CENTRAL1", true, null, "storage.us-central1.rep.googleapis.com" };
yield return new object[] { "US-CENTRAL1", true, "", "storage.us-central1.rep.googleapis.com" };
yield return new object[] { "US-CENTRAL1", true, " ", "storage.us-central1.rep.googleapis.com" };
yield return new object[] { "US-CENTRAL1", true, "null", "storage.us-central1.rep.googleapis.com" };
yield return new object[] { "US-CENTRAL1", true, "example.com", "example.com" };
}

private PutGetStageInfo CreateGcsStageInfo(string region, bool useRegionalUrl, string endPoint) =>
new PutGetStageInfo
{
locationType = SFRemoteStorageUtil.GCS_FS,
location = "some location",
path = "some path",
region = region,
storageAccount = "some storage account",
isClientSideEncrypted = true,
stageCredentials = new Dictionary<string, string>(),
presignedUrl = "some pre-signed url",
endPoint = endPoint,
useRegionalUrl = useRegionalUrl
};
}
}
35 changes: 34 additions & 1 deletion Snowflake.Data.Tests/UnitTests/SFAzureClientTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ namespace Snowflake.Data.Tests.UnitTests
using Azure;
using Azure.Storage.Blobs.Models;

[TestFixture]
[TestFixture, NonParallelizable]
class SFAzureClientTest : SFBaseTest
{
// Mock data for file metadata
Expand Down Expand Up @@ -377,5 +377,38 @@ public async Task TestDownloadFileAsync(HttpStatusCode httpStatusCode, ResultSta
// Assert
Assert.AreEqual(expectedResultStatus.ToString(), _fileMetadata.resultStatus);
}

[Test]
public void TestEncryptionMetadataReadingIsCaseInsensitive()
{
// arrange
var metadata = new Dictionary<string, string>
{
{
"ENCRYPTIONDATA",
@"{
""ContentEncryptionIV"": ""initVector"",
""WrappedContentKey"": {
""EncryptedKey"": ""key""
}
}"
},
{ "MATDESC", "description" },
{ "SFCDIGEST", "something"}
};
var blobProperties = BlobsModelFactory.BlobProperties(metadata: metadata, contentLength: 10);
var mockBlobServiceClient = new Mock<BlobServiceClient>();
_client = new SFSnowflakeAzureClient(_fileMetadata.stageInfo, mockBlobServiceClient.Object);

// act
var fileHeader = _client.HandleFileHeaderResponse(ref _fileMetadata, blobProperties);

// assert
Assert.AreEqual(ResultStatus.UPLOADED.ToString(), _fileMetadata.resultStatus);
Assert.AreEqual("something", fileHeader.digest);
Assert.AreEqual("initVector", fileHeader.encryptionMetadata.iv);
Assert.AreEqual("key", fileHeader.encryptionMetadata.key);
Assert.AreEqual("description", fileHeader.encryptionMetadata.matDesc);
}
}
}
2 changes: 2 additions & 0 deletions Snowflake.Data.Tests/UnitTests/SFEnvironmentTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,8 @@ public void TestRuntimeExtraction()
expectedVersion = "7.0";
#elif NET8_0
expectedVersion = "8.0";
#elif NET9_0
expectedVersion = "9.0";
#endif

// Act
Expand Down
98 changes: 84 additions & 14 deletions Snowflake.Data.Tests/UnitTests/SFGCSClientTest.cs
Original file line number Diff line number Diff line change
@@ -1,24 +1,25 @@
/*
* Copyright (c) 2012-2023 Snowflake Computing Inc. All rights reserved.
* Copyright (c) 2012-2024 Snowflake Computing Inc. All rights reserved.
*/

using System;
using NUnit.Framework;
using Snowflake.Data.Core;
using Snowflake.Data.Core.FileTransfer.StorageClient;
using Snowflake.Data.Core.FileTransfer;
using System.Collections.Generic;
using System.Net;
using System.IO;
using System.Linq;
using System.Net.Http;
using System.Threading.Tasks;
using System.Threading;
using Snowflake.Data.Tests.Mock;
using Moq;

namespace Snowflake.Data.Tests.UnitTests
{
using NUnit.Framework;
using Snowflake.Data.Core;
using Snowflake.Data.Core.FileTransfer.StorageClient;
using Snowflake.Data.Core.FileTransfer;
using System.Collections.Generic;
using System.Net;
using System.IO;
using System.Threading.Tasks;
using System.Threading;
using Snowflake.Data.Tests.Mock;
using Moq;

[TestFixture]
[TestFixture, NonParallelizable]
class SFGCSClientTest : SFBaseTest
{
// Mock data for file metadata
Expand Down Expand Up @@ -340,6 +341,75 @@ public async Task TestDownloadFileAsync(HttpStatusCode? httpStatusCode, ResultSt
AssertForDownloadFileTests(expectedResultStatus);
}

[Test]
[TestCase("us-central1", null, null, "https://storage.googleapis.com/mock-customer-stage/mock-id/tables/mock-key/")]
[TestCase("us-central1", "example.com", null, "https://example.com/mock-customer-stage/mock-id/tables/mock-key/")]
[TestCase("us-central1", "https://example.com", null, "https://example.com/mock-customer-stage/mock-id/tables/mock-key/")]
[TestCase("us-central1", null, true, "https://storage.us-central1.rep.googleapis.com/mock-customer-stage/mock-id/tables/mock-key/")]
[TestCase("me-central2", null, null, "https://storage.me-central2.rep.googleapis.com/mock-customer-stage/mock-id/tables/mock-key/")]
public void TestUseUriWithRegionsWhenNeeded(string region, string endPoint, bool useRegionalUrl, string expectedRequestUri)
{
var fileMetadata = new SFFileMetadata()
{
stageInfo = new PutGetStageInfo()
{
endPoint = endPoint,
location = Location,
locationType = SFRemoteStorageUtil.GCS_FS,
path = LocationPath,
presignedUrl = null,
region = region,
stageCredentials = _stageCredentials,
storageAccount = null,
useRegionalUrl = useRegionalUrl
}
};

// act
var uri = _client.FormBaseRequest(fileMetadata, "PUT").RequestUri.ToString();

// assert
Assert.AreEqual(expectedRequestUri, uri);
}

[Test]
[TestCase("some-header-name", "SOME-HEADER-NAME")]
[TestCase("SOME-HEADER-NAME", "some-header-name")]
public void TestGcsHeadersAreCaseInsensitiveForHttpResponseMessage(string headerNameToAdd, string headerNameToGet)
{
// arrange
const string HeaderValue = "someValue";
var responseMessage = new HttpResponseMessage( HttpStatusCode.OK ) {Content = new StringContent( "Response content" ) };
responseMessage.Headers.Add(headerNameToAdd, HeaderValue);

// act
var header = responseMessage.Headers.GetValues(headerNameToGet);

// assert
Assert.NotNull(header);
Assert.AreEqual(1, header.Count());
Assert.AreEqual(HeaderValue, header.First());
}

[Test]
[TestCase("some-header-name", "SOME-HEADER-NAME")]
[TestCase("SOME-HEADER-NAME", "some-header-name")]
public void TestGcsHeadersAreCaseInsensitiveForWebHeaderCollection(string headerNameToAdd, string headerNameToGet)
{
// arrange
const string HeaderValue = "someValue";
var headers = new WebHeaderCollection();
headers.Add(headerNameToAdd, HeaderValue);

// act
var header = headers.GetValues(headerNameToGet);

// assert
Assert.NotNull(header);
Assert.AreEqual(1, header.Count());
Assert.AreEqual(HeaderValue, header.First());
}

private void AssertForDownloadFileTests(ResultStatus expectedResultStatus)
{
if (expectedResultStatus == ResultStatus.DOWNLOADED)
Expand Down
55 changes: 38 additions & 17 deletions Snowflake.Data.Tests/UnitTests/SFS3ClientTest.cs
Original file line number Diff line number Diff line change
@@ -1,27 +1,25 @@
/*
* Copyright (c) 2012-2023 Snowflake Computing Inc. All rights reserved.
* Copyright (c) 2012-2024 Snowflake Computing Inc. All rights reserved.
*/

using System;
using Amazon.S3.Encryption;
using NUnit.Framework;
using Snowflake.Data.Core;
using Snowflake.Data.Core.FileTransfer.StorageClient;
using Snowflake.Data.Core.FileTransfer;
using System.Collections.Generic;
using Amazon.S3;
using Snowflake.Data.Tests.Mock;
using System.Threading.Tasks;
using Amazon;
using System.Threading;
using System.IO;
using Moq;
using Amazon.S3.Model;

namespace Snowflake.Data.Tests.UnitTests
{
using NUnit.Framework;
using Snowflake.Data.Core;
using Snowflake.Data.Core.FileTransfer.StorageClient;
using Snowflake.Data.Core.FileTransfer;
using System.Collections.Generic;
using Amazon.S3;
using Snowflake.Data.Tests.Mock;
using System.Threading.Tasks;
using Amazon;
using System.Threading;
using System.IO;
using Moq;
using Amazon.S3.Model;

[TestFixture]
[TestFixture, NonParallelizable]
class SFS3ClientTest : SFBaseTest
{
// Mock data for file metadata
Expand Down Expand Up @@ -320,6 +318,29 @@ public async Task TestDownloadFileAsync(string awsStatusCode, ResultStatus expec
AssertForDownloadFileTests(expectedResultStatus);
}

[Test]
public void TestEncryptionMetadataReadingIsCaseInsensitive()
{
// arrange
var mockAmazonS3Client = new Mock<AmazonS3Client>(AwsKeyId, AwsSecretKey, AwsToken, _clientConfig);
_client = new SFS3Client(_fileMetadata.stageInfo, MaxRetry, Parallel, _proxyCredentials, mockAmazonS3Client.Object);
var response = new GetObjectResponse();
response.Metadata.Add(SFS3Client.AMZ_IV.ToUpper(), "initVector");
response.Metadata.Add(SFS3Client.AMZ_KEY.ToUpper(), "key");
response.Metadata.Add(SFS3Client.AMZ_MATDESC.ToUpper(), "description");
response.Metadata.Add(SFS3Client.SFC_DIGEST.ToUpper(), "something");

// act
var fileHeader = _client.HandleFileHeaderResponse(ref _fileMetadata, response);

// assert
Assert.AreEqual(ResultStatus.UPLOADED.ToString(), _fileMetadata.resultStatus);
Assert.AreEqual("something", fileHeader.digest);
Assert.AreEqual("initVector", fileHeader.encryptionMetadata.iv);
Assert.AreEqual("key", fileHeader.encryptionMetadata.key);
Assert.AreEqual("description", fileHeader.encryptionMetadata.matDesc);
}

private void AssertForDownloadFileTests(ResultStatus expectedResultStatus)
{
if (expectedResultStatus == ResultStatus.DOWNLOADED)
Expand Down
Loading

0 comments on commit 26f52ee

Please sign in to comment.