Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SLVS-1433 Add credentials uri to the ServerConnection model #5660

Merged
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,239 @@
/*
* SonarLint for Visual Studio
* Copyright (C) 2016-2024 SonarSource SA
* mailto:info AT sonarsource DOT com
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 3 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/

using SonarLint.VisualStudio.ConnectedMode.Persistence;
using SonarLint.VisualStudio.Core.Binding;
using SonarLint.VisualStudio.TestInfrastructure;
using static SonarLint.VisualStudio.ConnectedMode.Persistence.ServerConnectionModelMapper;

namespace SonarLint.VisualStudio.ConnectedMode.UnitTests.Persistence;

[TestClass]
public class ServerConnectionModelMapperTest
{
private ServerConnectionModelMapper testSubject;

[TestInitialize]
public void TestInitialize()
{
testSubject = new ServerConnectionModelMapper();
}

[TestMethod]
public void MefCtor_CheckExports()
{
MefTestHelpers.CheckTypeCanBeImported<ServerConnectionModelMapper, IServerConnectionModelMapper>();
}

[TestMethod]
public void Mef_CheckIsSingleton()
{
MefTestHelpers.CheckIsSingletonMefComponent<ServerConnectionModelMapper>();
}

[TestMethod]
[DataRow(true)]
[DataRow(false)]
public void GetServerConnection_SonarCloud_ReturnsSonarCloudConnection(bool isSmartNotificationsEnabled)
{
var connectionsModel = GetSonarCloudJsonModel("myOrg", isSmartNotificationsEnabled);

var serverConnection = testSubject.GetServerConnection(connectionsModel);

IsExpectedSonarCloudConnection(serverConnection, connectionsModel);
}

[TestMethod]
[DataRow(true)]
[DataRow(false)]
public void GetServerConnection_SonarQube_ReturnsSonarQubeConnection(bool isSmartNotificationsEnabled)
{
var connectionsModel = GetSonarQubeJsonModel("http://localhost:9000", isSmartNotificationsEnabled);

var serverConnection = testSubject.GetServerConnection(connectionsModel);

IsExpectedSonarQubeConnection(serverConnection, connectionsModel);
}

[TestMethod]
public void GetServerConnection_BothOrganizationKeyAndServerUriAreSet_ThrowsException()
{
var connectionsModel = GetSonarCloudJsonModel("myOrg");
connectionsModel.ServerUri = "http://localhost:9000";

Action act = () => testSubject.GetServerConnection(connectionsModel);

act.Should().Throw<InvalidOperationException>($"Invalid {nameof(ServerConnectionJsonModel)}. {nameof(ServerConnection)} could not be created");
}

[TestMethod]
public void GetServerConnection_BothOrganizationKeyAndServerUriAreNull_ThrowsException()
{
var connectionsModel = GetSonarCloudJsonModel("myOrg");
connectionsModel.OrganizationKey = null;

Action act = () => testSubject.GetServerConnection(connectionsModel);

act.Should().Throw<InvalidOperationException>($"Invalid {nameof(ServerConnectionJsonModel)}. {nameof(ServerConnection)} could not be created");
}

[TestMethod]
[DataRow("org", null, true)]
[DataRow(null, "http://localhost", false)]
[DataRow(null, null, false)]
[DataRow("org", "http://localhost", false)]
public void IsServerConnectionForSonarCloud_OrganizationKeySetAndServerUriNotSet_ReturnsTrue(string organizationKey, string serverUi, bool expectedResult)
{
var connectionsModel = GetSonarCloudJsonModel(organizationKey);
connectionsModel.ServerUri = serverUi;

var isSonarCloud = IsServerConnectionForSonarCloud(connectionsModel);

isSonarCloud.Should().Be(expectedResult);
}

[TestMethod]
[DataRow("org", null, false)]
[DataRow(null, "http://localhost", true)]
[DataRow(null, null, false)]
[DataRow("org", "http://localhost", false)]
public void IsServerConnectionForSonarCloud_OrganizationKeyNotSetAndServerUriSet_ReturnsTrue(string organizationKey, string serverUi, bool expectedResult)
{
var connectionsModel = GetSonarCloudJsonModel(organizationKey);
connectionsModel.ServerUri = serverUi;

var isSonarQube = IsServerConnectionForSonarQube(connectionsModel);

isSonarQube.Should().Be(expectedResult);
}

[TestMethod]
[DataRow(true)]
[DataRow(false)]
public void GetServerConnectionsListJsonModel_OneSonarCloudConnection_ReturnsServerConnectionModelForSonarCloud(bool isSmartNotifications)
{
var sonarCloud = new ServerConnection.SonarCloud("myOrg", new ServerConnectionSettings(isSmartNotifications));

var model = testSubject.GetServerConnectionsListJsonModel([sonarCloud]);

model.Should().NotBeNull();
model.ServerConnections.Count.Should().Be(1);
model.ServerConnections[0].Should().BeEquivalentTo(new ServerConnectionJsonModel
{
Id = sonarCloud.Id,
OrganizationKey = sonarCloud.OrganizationKey,
Settings = sonarCloud.Settings,
ServerUri = null
});
}

[TestMethod]
public void GetServerConnectionsListJsonModel_OneSonarCloudConnectionWithNullSettings_ThrowsExceptions()
{
var sonarCloud = new ServerConnection.SonarCloud("myOrg")
{
Settings = null
};

Action act = () => testSubject.GetServerConnectionsListJsonModel([sonarCloud]);

act.Should().Throw<InvalidOperationException>($"{nameof(ServerConnection.Settings)} can not be null");
}

[TestMethod]
[DataRow(true)]
[DataRow(false)]
public void GetServerConnectionsListJsonModel_OneSonarQubeConnection_ReturnsServerConnectionModelForSonarQube(bool isSmartNotifications)
{
var sonarQube = new ServerConnection.SonarQube(new Uri("http://localhost"), new ServerConnectionSettings(isSmartNotifications));

var model = testSubject.GetServerConnectionsListJsonModel([sonarQube]);

model.Should().NotBeNull();
model.ServerConnections.Count.Should().Be(1);
model.ServerConnections[0].Should().BeEquivalentTo(new ServerConnectionJsonModel
{
Id = sonarQube.Id,
OrganizationKey = null,
ServerUri = sonarQube.ServerUri.ToString(),
Settings = sonarQube.Settings
});
}

[TestMethod]
public void GetServerConnectionsListJsonModel_OneSonarQubeConnectionWithNullSettings_ThrowsExceptions()
{
var sonarQube = new ServerConnection.SonarQube(new Uri("http://localhost"))
{
Settings = null
};

Action act = () => testSubject.GetServerConnectionsListJsonModel([sonarQube]);

act.Should().Throw<InvalidOperationException>($"{nameof(ServerConnection.Settings)} can not be null");
}

[TestMethod]
public void GetServerConnectionsListJsonModel_NoConnection_ReturnsServerConnectionModelWithNoConnection()
{
var model = testSubject.GetServerConnectionsListJsonModel([]);

model.Should().NotBeNull();
model.ServerConnections.Should().BeEmpty();
}

private static ServerConnectionJsonModel GetSonarCloudJsonModel(string id, bool isSmartNotificationsEnabled = false)
{
return new ServerConnectionJsonModel
{
Id = id,
OrganizationKey = id,
Settings = new ServerConnectionSettings(isSmartNotificationsEnabled)
};
}

private static ServerConnectionJsonModel GetSonarQubeJsonModel(string id, bool isSmartNotificationsEnabled = false)
{
return new ServerConnectionJsonModel
{
Id = id,
ServerUri = id,
Settings = new ServerConnectionSettings(isSmartNotificationsEnabled)
};
}

private static void IsExpectedSonarCloudConnection(ServerConnection serverConnection, ServerConnectionJsonModel connectionsModel)
{
serverConnection.Should().BeOfType<ServerConnection.SonarCloud>();
serverConnection.Id.Should().Be(serverConnection.Id);
serverConnection.Settings.Should().NotBeNull();
serverConnection.Settings.IsSmartNotificationsEnabled.Should().Be(connectionsModel.Settings.IsSmartNotificationsEnabled);
((ServerConnection.SonarCloud)serverConnection).OrganizationKey.Should().Be(connectionsModel.OrganizationKey);
}

private static void IsExpectedSonarQubeConnection(ServerConnection serverConnection, ServerConnectionJsonModel connectionsModel)
{
serverConnection.Should().BeOfType<ServerConnection.SonarQube>();
serverConnection.Id.Should().Be(serverConnection.Id);
serverConnection.Settings.Should().NotBeNull();
serverConnection.Settings.IsSmartNotificationsEnabled.Should().Be(connectionsModel.Settings.IsSmartNotificationsEnabled);
((ServerConnection.SonarQube)serverConnection).ServerUri.Should().Be(connectionsModel.ServerUri);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,8 @@
* Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/

using System;
using Microsoft.Alm.Authentication;
using SonarLint.VisualStudio.ConnectedMode.Binding;
using Moq;
using SonarLint.VisualStudio.ConnectedMode.Persistence;
using SonarLint.VisualStudio.Core.Binding;
using SonarQube.Client.Helpers;
Expand All @@ -31,16 +29,16 @@ namespace SonarLint.VisualStudio.ConnectedMode.UnitTests.Persistence
[TestClass]
public class SolutionBindingCredentialsLoaderTests
{
private Mock<ICredentialStoreService> store;
private ICredentialStoreService store;
private Uri mockUri;
private SolutionBindingCredentialsLoader testSubject;

[TestInitialize]
public void Setup()
{
store = new Mock<ICredentialStoreService>();
store = Substitute.For<ICredentialStoreService>();
mockUri = new Uri("http://sonarsource.com");
testSubject = new SolutionBindingCredentialsLoader(store.Object);
testSubject = new SolutionBindingCredentialsLoader(store);
}

[TestMethod]
Expand All @@ -61,7 +59,7 @@ public void Load_ServerUriIsNull_Null()
[TestMethod]
public void Load_NoCredentials_Null()
{
store.Setup(x => x.ReadCredentials(mockUri)).Returns(null as Credential);
store.ReadCredentials(mockUri).Returns(null as Credential);

var actual = testSubject.Load(mockUri);
actual.Should().Be(null);
Expand All @@ -72,7 +70,7 @@ public void Load_CredentialsExist_CredentialsWithSecuredString()
{
var credentials = new Credential("user", "password");
store
.Setup(x => x.ReadCredentials(It.Is<TargetUri>(t => t.ActualUri == mockUri)))
.ReadCredentials(Arg.Is<TargetUri>(t => t.ActualUri == mockUri))
.Returns(credentials);

var actual = testSubject.Load(mockUri);
Expand All @@ -86,15 +84,15 @@ public void Save_ServerUriIsNull_CredentialsNotSaved()

testSubject.Save(credentials, null);

store.Verify(x=> x.WriteCredentials(It.IsAny<TargetUri>(), It.IsAny<Credential>()), Times.Never);
store.DidNotReceive().WriteCredentials(Arg.Any<TargetUri>(), Arg.Any<Credential>());
}

[TestMethod]
public void Save_CredentialsAreNull_CredentialsNotSaved()
{
testSubject.Save(null, mockUri);

store.Verify(x => x.WriteCredentials(It.IsAny<TargetUri>(), It.IsAny<Credential>()), Times.Never);
store.DidNotReceive().WriteCredentials(Arg.Any<TargetUri>(), Arg.Any<Credential>());
}

[TestMethod]
Expand All @@ -103,7 +101,7 @@ public void Save_CredentialsAreNotBasicAuth_CredentialsNotSaved()
var mockCredentials = new Mock<ICredentials>();
testSubject.Save(mockCredentials.Object, mockUri);

store.Verify(x => x.WriteCredentials(It.IsAny<TargetUri>(), It.IsAny<Credential>()), Times.Never);
store.DidNotReceive().WriteCredentials(Arg.Any<TargetUri>(), Arg.Any<Credential>());
}

[TestMethod]
Expand All @@ -112,11 +110,26 @@ public void Save_CredentialsAreBasicAuth_CredentialsSavedWithUnsecuredString()
var credentials = new BasicAuthCredentials("user", "password".ToSecureString());
testSubject.Save(credentials, mockUri);

store.Verify(x =>
x.WriteCredentials(
It.Is<TargetUri>(t => t.ActualUri == mockUri),
It.Is<Credential>(c=> c.Username == "user" && c.Password == "password")),
Times.Once);
store.Received(1)
.WriteCredentials(
Arg.Is<TargetUri>(t => t.ActualUri == mockUri),
Arg.Is<Credential>(c=> c.Username == "user" && c.Password == "password"));
}

[TestMethod]
public void DeleteCredentials_UriNull_DoesNotCallStoreDeleteCredentials()
{
testSubject.DeleteCredentials(null);

store.DidNotReceive().DeleteCredentials(Arg.Any<TargetUri>());
}

[TestMethod]
public void DeleteCredentials_UriProvided_CallsStoreDeleteCredentials()
{
testSubject.DeleteCredentials(mockUri);

store.Received(1).DeleteCredentials(Arg.Any<TargetUri>());
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ namespace SonarLint.VisualStudio.ConnectedMode.Persistence
{
interface ISolutionBindingCredentialsLoader
{
void DeleteCredentials(Uri boundServerUri);
ICredentials Load(Uri boundServerUri);
void Save(ICredentials credentials, Uri boundServerUri);
}
Expand Down
Loading