From 7d490933dd627e460c6cf4e04fcb1d2fd74b3b46 Mon Sep 17 00:00:00 2001 From: Gabriela Trutan Date: Wed, 4 Dec 2024 15:01:54 +0100 Subject: [PATCH 1/5] SLVS-1404 CaYC: replace Moq --- .../SolutionBindingFileLoaderTests.cs | 55 +++++++++---------- 1 file changed, 26 insertions(+), 29 deletions(-) diff --git a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs index 5912c344e..6310ecf31 100644 --- a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs +++ b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs @@ -20,6 +20,7 @@ using System.IO; using System.IO.Abstractions; +using NSubstitute.ExceptionExtensions; using SonarLint.VisualStudio.ConnectedMode.Persistence; using SonarLint.VisualStudio.Core; using SonarLint.VisualStudio.Core.Binding; @@ -29,8 +30,8 @@ namespace SonarLint.VisualStudio.ConnectedMode.UnitTests.Persistence [TestClass] public class SolutionBindingFileLoaderTests { - private Mock logger; - private Mock fileSystem; + private ILogger logger; + private IFileSystem fileSystem; private SolutionBindingFileLoader testSubject; private BindingJsonModel bindingJsonModel; private string serializedProject; @@ -41,12 +42,12 @@ public class SolutionBindingFileLoaderTests [TestInitialize] public void TestInitialize() { - logger = new Mock(); - fileSystem = new Mock(); + logger = Substitute.For(); + fileSystem = Substitute.For(); - testSubject = new SolutionBindingFileLoader(logger.Object, fileSystem.Object); + testSubject = new SolutionBindingFileLoader(logger, fileSystem); - fileSystem.Setup(x => x.Directory.Exists(MockDirectory)).Returns(true); + fileSystem.Directory.Exists(MockDirectory).Returns(true); bindingJsonModel = new BindingJsonModel { @@ -91,7 +92,7 @@ public void Ctor_NullLogger_Exception() [TestMethod] public void Ctor_NullFileSystem_Exception() { - Action act = () => new SolutionBindingFileLoader(logger.Object, null); + Action act = () => new SolutionBindingFileLoader(logger, null); act.Should().ThrowExactly().And.ParamName.Should().Be("fileSystem"); } @@ -99,28 +100,26 @@ public void Ctor_NullFileSystem_Exception() [TestMethod] public void Save_DirectoryDoesNotExist_DirectoryIsCreated() { - fileSystem.Setup(x => x.Directory.Exists(MockDirectory)).Returns(false); + fileSystem.Directory.Exists(MockDirectory).Returns(false); testSubject.Save(MockFilePath, bindingJsonModel); - fileSystem.Verify(x => x.Directory.CreateDirectory(MockDirectory), Times.Once); + fileSystem.Directory.Received(1).CreateDirectory(MockDirectory); } [TestMethod] public void Save_DirectoryExists_DirectoryNotCreated() { - fileSystem.Setup(x => x.Directory.Exists(MockDirectory)).Returns(true); + fileSystem.Directory.Exists(MockDirectory).Returns(true); testSubject.Save(MockFilePath, bindingJsonModel); - fileSystem.Verify(x => x.Directory.CreateDirectory(It.IsAny()), Times.Never); + fileSystem.Directory.DidNotReceive().CreateDirectory(Arg.Any()); } [TestMethod] public void Save_ReturnsTrue() { - fileSystem.Setup(x => x.File.WriteAllText(MockFilePath, serializedProject)); - var actual = testSubject.Save(MockFilePath, bindingJsonModel); actual.Should().BeTrue(); } @@ -128,17 +127,15 @@ public void Save_ReturnsTrue() [TestMethod] public void Save_FileSerializedAndWritten() { - fileSystem.Setup(x => x.File.WriteAllText(MockFilePath, serializedProject)); - testSubject.Save(MockFilePath, bindingJsonModel); - fileSystem.Verify(x => x.File.WriteAllText(MockFilePath, serializedProject), Times.Once); + fileSystem.File.Received(1).WriteAllText(MockFilePath, serializedProject); } [TestMethod] public void Save_NonCriticalException_False() { - fileSystem.Setup(x => x.File.WriteAllText(MockFilePath, It.IsAny())).Throws(); + fileSystem.File.When(x => x.WriteAllText(MockFilePath, Arg.Any())).Throw(); var actual = testSubject.Save(MockFilePath, bindingJsonModel); actual.Should().BeFalse(); @@ -147,7 +144,7 @@ public void Save_NonCriticalException_False() [TestMethod] public void Save_CriticalException_Exception() { - fileSystem.Setup(x => x.File.WriteAllText(MockFilePath, It.IsAny())).Throws(); + fileSystem.File.When(x => x.WriteAllText(MockFilePath, Arg.Any())).Throw(); Action act = () => testSubject.Save(MockFilePath, bindingJsonModel); @@ -166,7 +163,7 @@ public void Load_FilePathIsNull_Null(string filePath) [TestMethod] public void Load_FileDoesNotExist_Null() { - fileSystem.Setup(x => x.File.Exists(MockFilePath)).Returns(false); + fileSystem.File.Exists(MockFilePath).Returns(false); var actual = testSubject.Load(MockFilePath); actual.Should().Be(null); @@ -175,8 +172,8 @@ public void Load_FileDoesNotExist_Null() [TestMethod] public void Load_InvalidJson_Null() { - fileSystem.Setup(x => x.File.Exists(MockFilePath)).Returns(true); - fileSystem.Setup(x => x.File.ReadAllText(MockFilePath)).Returns("bad json"); + fileSystem.File.Exists(MockFilePath).Returns(true); + fileSystem.File.ReadAllText(MockFilePath).Returns("bad json"); var actual = testSubject.Load(MockFilePath); actual.Should().Be(null); @@ -185,8 +182,8 @@ public void Load_InvalidJson_Null() [TestMethod] public void Load_NonCriticalException_Null() { - fileSystem.Setup(x => x.File.Exists(MockFilePath)).Returns(true); - fileSystem.Setup(x => x.File.ReadAllText(MockFilePath)).Throws(); + fileSystem.File.Exists(MockFilePath).Returns(true); + fileSystem.File.ReadAllText(MockFilePath).Throws(); var actual = testSubject.Load(MockFilePath); actual.Should().Be(null); @@ -195,8 +192,8 @@ public void Load_NonCriticalException_Null() [TestMethod] public void Load_CriticalException_Exception() { - fileSystem.Setup(x => x.File.Exists(MockFilePath)).Returns(true); - fileSystem.Setup(x => x.File.ReadAllText(MockFilePath)).Throws(); + fileSystem.File.Exists(MockFilePath).Returns(true); + fileSystem.File.ReadAllText(MockFilePath).Throws(); Action act = () => testSubject.Load(MockFilePath); @@ -206,8 +203,8 @@ public void Load_CriticalException_Exception() [TestMethod] public void Load_FileExists_DeserializedProject() { - fileSystem.Setup(x => x.File.Exists(MockFilePath)).Returns(true); - fileSystem.Setup(x => x.File.ReadAllText(MockFilePath)).Returns(serializedProject); + fileSystem.File.Exists(MockFilePath).Returns(true); + fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); var actual = testSubject.Load(MockFilePath); actual.Should().BeEquivalentTo(bindingJsonModel); @@ -220,8 +217,8 @@ public void Load_FileExists_ProjectWithNonUtcTimestamp_DeserializedProjectWithCo const string localDate = "2020-02-25T10:57:54+02:00"; serializedProject = serializedProject.Replace(utcDate, localDate); - fileSystem.Setup(x => x.File.Exists(MockFilePath)).Returns(true); - fileSystem.Setup(x => x.File.ReadAllText(MockFilePath)).Returns(serializedProject); + fileSystem.File.Exists(MockFilePath).Returns(true); + fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); var actual = testSubject.Load(MockFilePath); actual.Should().BeEquivalentTo(bindingJsonModel); From 12747e63ef712f4557220e9455b5043c203deb7a Mon Sep 17 00:00:00 2001 From: Gabriela Trutan Date: Thu, 5 Dec 2024 08:47:14 +0100 Subject: [PATCH 2/5] SLVS-1404 Implement the deletion of the binding directory. --- .../SolutionBindingFileLoaderTests.cs | 53 ++++++++++++++++--- .../Persistence/ISolutionBindingFileLoader.cs | 1 + .../PersistenceStrings.Designer.cs | 9 ++++ .../Persistence/PersistenceStrings.resx | 3 ++ .../Persistence/SolutionBindingFileLoader.cs | 15 ++++++ 5 files changed, 75 insertions(+), 6 deletions(-) diff --git a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs index 6310ecf31..999af05b6 100644 --- a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs +++ b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs @@ -20,6 +20,7 @@ using System.IO; using System.IO.Abstractions; +using NSubstitute; using NSubstitute.ExceptionExtensions; using SonarLint.VisualStudio.ConnectedMode.Persistence; using SonarLint.VisualStudio.Core; @@ -163,7 +164,7 @@ public void Load_FilePathIsNull_Null(string filePath) [TestMethod] public void Load_FileDoesNotExist_Null() { - fileSystem.File.Exists(MockFilePath).Returns(false); + MockFileNotExists(MockFilePath); var actual = testSubject.Load(MockFilePath); actual.Should().Be(null); @@ -172,7 +173,7 @@ public void Load_FileDoesNotExist_Null() [TestMethod] public void Load_InvalidJson_Null() { - fileSystem.File.Exists(MockFilePath).Returns(true); + MockFileExists(MockFilePath); fileSystem.File.ReadAllText(MockFilePath).Returns("bad json"); var actual = testSubject.Load(MockFilePath); @@ -182,7 +183,7 @@ public void Load_InvalidJson_Null() [TestMethod] public void Load_NonCriticalException_Null() { - fileSystem.File.Exists(MockFilePath).Returns(true); + MockFileExists(MockFilePath); fileSystem.File.ReadAllText(MockFilePath).Throws(); var actual = testSubject.Load(MockFilePath); @@ -192,7 +193,7 @@ public void Load_NonCriticalException_Null() [TestMethod] public void Load_CriticalException_Exception() { - fileSystem.File.Exists(MockFilePath).Returns(true); + MockFileExists(MockFilePath); fileSystem.File.ReadAllText(MockFilePath).Throws(); Action act = () => testSubject.Load(MockFilePath); @@ -203,7 +204,7 @@ public void Load_CriticalException_Exception() [TestMethod] public void Load_FileExists_DeserializedProject() { - fileSystem.File.Exists(MockFilePath).Returns(true); + MockFileExists(MockFilePath); fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); var actual = testSubject.Load(MockFilePath); @@ -217,7 +218,7 @@ public void Load_FileExists_ProjectWithNonUtcTimestamp_DeserializedProjectWithCo const string localDate = "2020-02-25T10:57:54+02:00"; serializedProject = serializedProject.Replace(utcDate, localDate); - fileSystem.File.Exists(MockFilePath).Returns(true); + MockFileExists(MockFilePath); fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); var actual = testSubject.Load(MockFilePath); @@ -226,5 +227,45 @@ public void Load_FileExists_ProjectWithNonUtcTimestamp_DeserializedProjectWithCo var deserializedTimestamp = actual.Profiles[Language.CSharp].ProfileTimestamp.Value.ToUniversalTime(); deserializedTimestamp.Should().Be(new DateTime(2020, 2, 25, 8, 57, 54)); } + + [TestMethod] + public void DeleteBindingDirectory_ConfigFilePathNotExists_ReturnsFalseAndLogs() + { + MockFileNotExists(MockFilePath); + + var result = testSubject.DeleteBindingDirectory(MockFilePath); + + result.Should().BeFalse(); + fileSystem.Directory.DidNotReceive().Delete(MockDirectory, recursive:true); + logger.Received(1).LogVerbose(PersistenceStrings.BindingDirectoryNotDeleted, MockFilePath); + } + + [TestMethod] + public void DeleteBindingDirectory_ConfigFilePathExists_DeletesBindingDirectoryRecursively() + { + MockFileExists(MockFilePath); + + var result = testSubject.DeleteBindingDirectory(MockFilePath); + + result.Should().BeTrue(); + fileSystem.Directory.Received(1).Delete(MockDirectory, recursive:true); + } + + [TestMethod] + public void DeleteBindingDirectory_DeletingDirectoryThrows_ReturnsFalseAndLogs() + { + MockFileExists(MockFilePath); + fileSystem.Directory.When(x => x.Delete(MockDirectory, recursive: true)).Throw(); + + var result = testSubject.DeleteBindingDirectory(MockFilePath); + + result.Should().BeFalse(); + fileSystem.Directory.Received(1).Delete(MockDirectory, recursive: true); + logger.Received(1).WriteLine(Arg.Any()); + } + + private void MockFileExists(string filePath) => fileSystem.File.Exists(filePath).Returns(true); + + private void MockFileNotExists(string filePath) => fileSystem.File.Exists(filePath).Returns(false); } } diff --git a/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs b/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs index 69e70a2ea..63224a026 100644 --- a/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs +++ b/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs @@ -24,5 +24,6 @@ internal interface ISolutionBindingFileLoader { BindingJsonModel Load(string filePath); bool Save(string filePath, BindingJsonModel project); + bool DeleteBindingDirectory(string configFilePath); } } diff --git a/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs b/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs index adf1f9ea5..b1ebadff7 100644 --- a/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs +++ b/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs @@ -60,6 +60,15 @@ internal PersistenceStrings() { } } + /// + /// Looks up a localized string similar to The binding configuration file '{0}' does not exist. Binding directory not deleted.. + /// + internal static string BindingDirectoryNotDeleted { + get { + return ResourceManager.GetString("BindingDirectoryNotDeleted", resourceCulture); + } + } + /// /// Looks up a localized string similar to Failed loading file: '{0}'. Please rebind solution to create a new configuration file.. /// diff --git a/src/ConnectedMode/Persistence/PersistenceStrings.resx b/src/ConnectedMode/Persistence/PersistenceStrings.resx index 682885485..4758ee42a 100644 --- a/src/ConnectedMode/Persistence/PersistenceStrings.resx +++ b/src/ConnectedMode/Persistence/PersistenceStrings.resx @@ -121,4 +121,7 @@ Failed loading file: '{0}'. Please rebind solution to create a new configuration file. Output window message in case of deserialization error. {0} file path. + + The binding configuration file '{0}' does not exist. Binding directory not deleted. + \ No newline at end of file diff --git a/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs b/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs index 98953b5a0..2ec1c1478 100644 --- a/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs +++ b/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs @@ -51,6 +51,21 @@ public bool Save(string filePath, BindingJsonModel project) return SafePerformFileSystemOperation(() => WriteConfig(filePath, serializedProject)); } + public bool DeleteBindingDirectory(string configFilePath) + { + if (fileSystem.File.Exists(configFilePath)) + { + return SafePerformFileSystemOperation(() => + { + var directoryName = Path.GetDirectoryName(configFilePath); + fileSystem.Directory.Delete(directoryName, recursive:true); + }); + } + + logger.LogVerbose(PersistenceStrings.BindingDirectoryNotDeleted, configFilePath); + return false; + } + private void WriteConfig(string configFile, string serializedProject) { Debug.Assert(!string.IsNullOrWhiteSpace(configFile)); From 2a1f8d0da76cb017a9482e3b007d7da2e94984f8 Mon Sep 17 00:00:00 2001 From: Gabriela Trutan Date: Thu, 5 Dec 2024 09:41:02 +0100 Subject: [PATCH 3/5] SLVS-1404 Implement DeleteBinding method in SolutionBindingRepository --- .../SolutionBindingRepositoryTests.cs | 24 +++++++++++++++++++ .../Persistence/SolutionBindingRepository.cs | 6 ++++- .../Binding/ISolutionBindingRepository.cs | 3 ++- 3 files changed, 31 insertions(+), 2 deletions(-) diff --git a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs index 68a46629b..6de8ea999 100644 --- a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs +++ b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs @@ -44,6 +44,7 @@ public class SolutionBindingRepositoryTests private BasicAuthCredentials mockCredentials; private const string MockFilePath = "test file path"; + private const string LocalBindingKey = "my solution"; [TestInitialize] public void TestInitialize() @@ -291,6 +292,29 @@ public void LegacyRead_ValidBinding_LoadsCredentials() credentialsLoader.Received().Load(bindingJsonModel.ServerUri); } + [TestMethod] + public void DeleteBinding_DeletesBindingDirectoryOfBindingFile() + { + unintrusiveBindingPathProvider.GetBindingPath(LocalBindingKey).Returns(MockFilePath); + + testSubject.DeleteBinding(LocalBindingKey); + + solutionBindingFileLoader.Received(1).DeleteBindingDirectory(MockFilePath); + } + + [TestMethod] + [DataRow(true)] + [DataRow(false)] + public void DeleteBinding_ReturnsResultOfDeleteBindingDirectory(bool expectedResult) + { + unintrusiveBindingPathProvider.GetBindingPath(LocalBindingKey).Returns(MockFilePath); + solutionBindingFileLoader.DeleteBindingDirectory(MockFilePath).Returns(expectedResult); + + var result = testSubject.DeleteBinding(LocalBindingKey); + + result.Should().Be(expectedResult); + } + private BoundServerProject SetUpBinding(string solution, ServerConnection connection, string bindingConfig) { var dto = new BindingJsonModel{ServerConnectionId = connection?.Id}; diff --git a/src/ConnectedMode/Persistence/SolutionBindingRepository.cs b/src/ConnectedMode/Persistence/SolutionBindingRepository.cs index 79b22f5c1..e67db54d4 100644 --- a/src/ConnectedMode/Persistence/SolutionBindingRepository.cs +++ b/src/ConnectedMode/Persistence/SolutionBindingRepository.cs @@ -100,7 +100,11 @@ public bool Write(string configFilePath, BoundServerProject binding) return true; } - public bool DeleteBinding(string configFilePath) => throw new NotImplementedException(); + public bool DeleteBinding(string localBindingKey) + { + var bindingPath = unintrusiveBindingPathProvider.GetBindingPath(localBindingKey); + return solutionBindingFileLoader.DeleteBindingDirectory(bindingPath); + } public event EventHandler BindingUpdated; diff --git a/src/Core/Binding/ISolutionBindingRepository.cs b/src/Core/Binding/ISolutionBindingRepository.cs index 99d63a458..ca63684c6 100644 --- a/src/Core/Binding/ISolutionBindingRepository.cs +++ b/src/Core/Binding/ISolutionBindingRepository.cs @@ -46,8 +46,9 @@ public interface ISolutionBindingRepository /// /// Deletes the binding information /// + /// The local binding key of the /// If binding has been deleted - bool DeleteBinding(string configFilePath); + bool DeleteBinding(string localBindingKey); /// /// Raises when operation completes successfully From 2675c02d24966ad523b41d991fe42cf2af6f799f Mon Sep 17 00:00:00 2001 From: Gabriela Trutan Date: Thu, 5 Dec 2024 09:58:48 +0100 Subject: [PATCH 4/5] SLVS-1404 Apply code cleanup in touched files --- .../SolutionBindingFileLoaderTests.cs | 355 +++++++++--------- .../SolutionBindingRepositoryTests.cs | 95 ++--- .../Persistence/ISolutionBindingFileLoader.cs | 15 +- .../Persistence/SolutionBindingFileLoader.cs | 163 ++++---- .../Binding/ISolutionBindingRepository.cs | 2 +- 5 files changed, 296 insertions(+), 334 deletions(-) diff --git a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs index 999af05b6..862a5988b 100644 --- a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs +++ b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingFileLoaderTests.cs @@ -20,56 +20,48 @@ using System.IO; using System.IO.Abstractions; -using NSubstitute; using NSubstitute.ExceptionExtensions; using SonarLint.VisualStudio.ConnectedMode.Persistence; using SonarLint.VisualStudio.Core; using SonarLint.VisualStudio.Core.Binding; -namespace SonarLint.VisualStudio.ConnectedMode.UnitTests.Persistence +namespace SonarLint.VisualStudio.ConnectedMode.UnitTests.Persistence; + +[TestClass] +public class SolutionBindingFileLoaderTests { - [TestClass] - public class SolutionBindingFileLoaderTests + private const string MockFilePath = "c:\\test.txt"; + private const string MockDirectory = "c:\\"; + private BindingJsonModel bindingJsonModel; + private IFileSystem fileSystem; + private ILogger logger; + private string serializedProject; + private SolutionBindingFileLoader testSubject; + + [TestInitialize] + public void TestInitialize() { - private ILogger logger; - private IFileSystem fileSystem; - private SolutionBindingFileLoader testSubject; - private BindingJsonModel bindingJsonModel; - private string serializedProject; - - private const string MockFilePath = "c:\\test.txt"; - private const string MockDirectory = "c:\\"; + logger = Substitute.For(); + fileSystem = Substitute.For(); - [TestInitialize] - public void TestInitialize() - { - logger = Substitute.For(); - fileSystem = Substitute.For(); + testSubject = new SolutionBindingFileLoader(logger, fileSystem); - testSubject = new SolutionBindingFileLoader(logger, fileSystem); + fileSystem.Directory.Exists(MockDirectory).Returns(true); - fileSystem.Directory.Exists(MockDirectory).Returns(true); - - bindingJsonModel = new BindingJsonModel + bindingJsonModel = new BindingJsonModel + { + ServerUri = new Uri("http://xxx.www.zzz/yyy:9000"), + Organization = null, + ProjectKey = "MyProject Key", + ProjectName = "projectName", + ServerConnectionId = null, + Profiles = new Dictionary { - ServerUri = new Uri("http://xxx.www.zzz/yyy:9000"), - Organization = null, - ProjectKey = "MyProject Key", - ProjectName = "projectName", - ServerConnectionId = null, - Profiles = new Dictionary - { - { - Language.CSharp, - new ApplicableQualityProfile - { - ProfileKey = "sonar way", ProfileTimestamp = DateTime.Parse("2020-02-25T08:57:54+0000") - } - } - } - }; - - serializedProject = @"{ + { Language.CSharp, new ApplicableQualityProfile { ProfileKey = "sonar way", ProfileTimestamp = DateTime.Parse("2020-02-25T08:57:54+0000") } } + } + }; + + serializedProject = @"{ ""ServerUri"": ""http://xxx.www.zzz/yyy:9000"", ""ProjectKey"": ""MyProject Key"", ""ProjectName"": ""projectName"", @@ -80,192 +72,191 @@ public void TestInitialize() } } }"; - } + } - [TestMethod] - public void Ctor_NullLogger_Exception() - { - Action act = () => new SolutionBindingFileLoader(null, null); + [TestMethod] + public void Ctor_NullLogger_Exception() + { + Action act = () => new SolutionBindingFileLoader(null, null); - act.Should().ThrowExactly().And.ParamName.Should().Be("logger"); - } + act.Should().ThrowExactly().And.ParamName.Should().Be("logger"); + } - [TestMethod] - public void Ctor_NullFileSystem_Exception() - { - Action act = () => new SolutionBindingFileLoader(logger, null); + [TestMethod] + public void Ctor_NullFileSystem_Exception() + { + Action act = () => new SolutionBindingFileLoader(logger, null); - act.Should().ThrowExactly().And.ParamName.Should().Be("fileSystem"); - } + act.Should().ThrowExactly().And.ParamName.Should().Be("fileSystem"); + } - [TestMethod] - public void Save_DirectoryDoesNotExist_DirectoryIsCreated() - { - fileSystem.Directory.Exists(MockDirectory).Returns(false); + [TestMethod] + public void Save_DirectoryDoesNotExist_DirectoryIsCreated() + { + fileSystem.Directory.Exists(MockDirectory).Returns(false); - testSubject.Save(MockFilePath, bindingJsonModel); + testSubject.Save(MockFilePath, bindingJsonModel); - fileSystem.Directory.Received(1).CreateDirectory(MockDirectory); - } + fileSystem.Directory.Received(1).CreateDirectory(MockDirectory); + } - [TestMethod] - public void Save_DirectoryExists_DirectoryNotCreated() - { - fileSystem.Directory.Exists(MockDirectory).Returns(true); + [TestMethod] + public void Save_DirectoryExists_DirectoryNotCreated() + { + fileSystem.Directory.Exists(MockDirectory).Returns(true); - testSubject.Save(MockFilePath, bindingJsonModel); + testSubject.Save(MockFilePath, bindingJsonModel); - fileSystem.Directory.DidNotReceive().CreateDirectory(Arg.Any()); - } + fileSystem.Directory.DidNotReceive().CreateDirectory(Arg.Any()); + } - [TestMethod] - public void Save_ReturnsTrue() - { - var actual = testSubject.Save(MockFilePath, bindingJsonModel); - actual.Should().BeTrue(); - } + [TestMethod] + public void Save_ReturnsTrue() + { + var actual = testSubject.Save(MockFilePath, bindingJsonModel); + actual.Should().BeTrue(); + } - [TestMethod] - public void Save_FileSerializedAndWritten() - { - testSubject.Save(MockFilePath, bindingJsonModel); + [TestMethod] + public void Save_FileSerializedAndWritten() + { + testSubject.Save(MockFilePath, bindingJsonModel); - fileSystem.File.Received(1).WriteAllText(MockFilePath, serializedProject); - } + fileSystem.File.Received(1).WriteAllText(MockFilePath, serializedProject); + } - [TestMethod] - public void Save_NonCriticalException_False() - { - fileSystem.File.When(x => x.WriteAllText(MockFilePath, Arg.Any())).Throw(); + [TestMethod] + public void Save_NonCriticalException_False() + { + fileSystem.File.When(x => x.WriteAllText(MockFilePath, Arg.Any())).Throw(); - var actual = testSubject.Save(MockFilePath, bindingJsonModel); - actual.Should().BeFalse(); - } + var actual = testSubject.Save(MockFilePath, bindingJsonModel); + actual.Should().BeFalse(); + } - [TestMethod] - public void Save_CriticalException_Exception() - { - fileSystem.File.When(x => x.WriteAllText(MockFilePath, Arg.Any())).Throw(); + [TestMethod] + public void Save_CriticalException_Exception() + { + fileSystem.File.When(x => x.WriteAllText(MockFilePath, Arg.Any())).Throw(); - Action act = () => testSubject.Save(MockFilePath, bindingJsonModel); + Action act = () => testSubject.Save(MockFilePath, bindingJsonModel); - act.Should().ThrowExactly(); - } + act.Should().ThrowExactly(); + } - [DataTestMethod] - [DataRow("")] - [DataRow(null)] - public void Load_FilePathIsNull_Null(string filePath) - { - var actual = testSubject.Load(filePath); - actual.Should().Be(null); - } + [DataTestMethod] + [DataRow("")] + [DataRow(null)] + public void Load_FilePathIsNull_Null(string filePath) + { + var actual = testSubject.Load(filePath); + actual.Should().Be(null); + } - [TestMethod] - public void Load_FileDoesNotExist_Null() - { - MockFileNotExists(MockFilePath); + [TestMethod] + public void Load_FileDoesNotExist_Null() + { + MockFileNotExists(MockFilePath); - var actual = testSubject.Load(MockFilePath); - actual.Should().Be(null); - } + var actual = testSubject.Load(MockFilePath); + actual.Should().Be(null); + } - [TestMethod] - public void Load_InvalidJson_Null() - { - MockFileExists(MockFilePath); - fileSystem.File.ReadAllText(MockFilePath).Returns("bad json"); + [TestMethod] + public void Load_InvalidJson_Null() + { + MockFileExists(MockFilePath); + fileSystem.File.ReadAllText(MockFilePath).Returns("bad json"); - var actual = testSubject.Load(MockFilePath); - actual.Should().Be(null); - } + var actual = testSubject.Load(MockFilePath); + actual.Should().Be(null); + } - [TestMethod] - public void Load_NonCriticalException_Null() - { - MockFileExists(MockFilePath); - fileSystem.File.ReadAllText(MockFilePath).Throws(); + [TestMethod] + public void Load_NonCriticalException_Null() + { + MockFileExists(MockFilePath); + fileSystem.File.ReadAllText(MockFilePath).Throws(); - var actual = testSubject.Load(MockFilePath); - actual.Should().Be(null); - } + var actual = testSubject.Load(MockFilePath); + actual.Should().Be(null); + } - [TestMethod] - public void Load_CriticalException_Exception() - { - MockFileExists(MockFilePath); - fileSystem.File.ReadAllText(MockFilePath).Throws(); + [TestMethod] + public void Load_CriticalException_Exception() + { + MockFileExists(MockFilePath); + fileSystem.File.ReadAllText(MockFilePath).Throws(); - Action act = () => testSubject.Load(MockFilePath); + Action act = () => testSubject.Load(MockFilePath); - act.Should().ThrowExactly(); - } + act.Should().ThrowExactly(); + } - [TestMethod] - public void Load_FileExists_DeserializedProject() - { - MockFileExists(MockFilePath); - fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); + [TestMethod] + public void Load_FileExists_DeserializedProject() + { + MockFileExists(MockFilePath); + fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); - var actual = testSubject.Load(MockFilePath); - actual.Should().BeEquivalentTo(bindingJsonModel); - } + var actual = testSubject.Load(MockFilePath); + actual.Should().BeEquivalentTo(bindingJsonModel); + } - [TestMethod] - public void Load_FileExists_ProjectWithNonUtcTimestamp_DeserializedProjectWithCorrectTimestampData() - { - const string utcDate = "2020-02-25T08:57:54Z"; - const string localDate = "2020-02-25T10:57:54+02:00"; - serializedProject = serializedProject.Replace(utcDate, localDate); + [TestMethod] + public void Load_FileExists_ProjectWithNonUtcTimestamp_DeserializedProjectWithCorrectTimestampData() + { + const string utcDate = "2020-02-25T08:57:54Z"; + const string localDate = "2020-02-25T10:57:54+02:00"; + serializedProject = serializedProject.Replace(utcDate, localDate); - MockFileExists(MockFilePath); - fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); + MockFileExists(MockFilePath); + fileSystem.File.ReadAllText(MockFilePath).Returns(serializedProject); - var actual = testSubject.Load(MockFilePath); - actual.Should().BeEquivalentTo(bindingJsonModel); + var actual = testSubject.Load(MockFilePath); + actual.Should().BeEquivalentTo(bindingJsonModel); - var deserializedTimestamp = actual.Profiles[Language.CSharp].ProfileTimestamp.Value.ToUniversalTime(); - deserializedTimestamp.Should().Be(new DateTime(2020, 2, 25, 8, 57, 54)); - } + var deserializedTimestamp = actual.Profiles[Language.CSharp].ProfileTimestamp.Value.ToUniversalTime(); + deserializedTimestamp.Should().Be(new DateTime(2020, 2, 25, 8, 57, 54)); + } - [TestMethod] - public void DeleteBindingDirectory_ConfigFilePathNotExists_ReturnsFalseAndLogs() - { - MockFileNotExists(MockFilePath); + [TestMethod] + public void DeleteBindingDirectory_ConfigFilePathNotExists_ReturnsFalseAndLogs() + { + MockFileNotExists(MockFilePath); - var result = testSubject.DeleteBindingDirectory(MockFilePath); + var result = testSubject.DeleteBindingDirectory(MockFilePath); - result.Should().BeFalse(); - fileSystem.Directory.DidNotReceive().Delete(MockDirectory, recursive:true); - logger.Received(1).LogVerbose(PersistenceStrings.BindingDirectoryNotDeleted, MockFilePath); - } + result.Should().BeFalse(); + fileSystem.Directory.DidNotReceive().Delete(MockDirectory, true); + logger.Received(1).LogVerbose(PersistenceStrings.BindingDirectoryNotDeleted, MockFilePath); + } - [TestMethod] - public void DeleteBindingDirectory_ConfigFilePathExists_DeletesBindingDirectoryRecursively() - { - MockFileExists(MockFilePath); + [TestMethod] + public void DeleteBindingDirectory_ConfigFilePathExists_DeletesBindingDirectoryRecursively() + { + MockFileExists(MockFilePath); - var result = testSubject.DeleteBindingDirectory(MockFilePath); + var result = testSubject.DeleteBindingDirectory(MockFilePath); - result.Should().BeTrue(); - fileSystem.Directory.Received(1).Delete(MockDirectory, recursive:true); - } + result.Should().BeTrue(); + fileSystem.Directory.Received(1).Delete(MockDirectory, true); + } - [TestMethod] - public void DeleteBindingDirectory_DeletingDirectoryThrows_ReturnsFalseAndLogs() - { - MockFileExists(MockFilePath); - fileSystem.Directory.When(x => x.Delete(MockDirectory, recursive: true)).Throw(); + [TestMethod] + public void DeleteBindingDirectory_DeletingDirectoryThrows_ReturnsFalseAndLogs() + { + MockFileExists(MockFilePath); + fileSystem.Directory.When(x => x.Delete(MockDirectory, true)).Throw(); - var result = testSubject.DeleteBindingDirectory(MockFilePath); + var result = testSubject.DeleteBindingDirectory(MockFilePath); - result.Should().BeFalse(); - fileSystem.Directory.Received(1).Delete(MockDirectory, recursive: true); - logger.Received(1).WriteLine(Arg.Any()); - } + result.Should().BeFalse(); + fileSystem.Directory.Received(1).Delete(MockDirectory, true); + logger.Received(1).WriteLine(Arg.Any()); + } - private void MockFileExists(string filePath) => fileSystem.File.Exists(filePath).Returns(true); + private void MockFileExists(string filePath) => fileSystem.File.Exists(filePath).Returns(true); - private void MockFileNotExists(string filePath) => fileSystem.File.Exists(filePath).Returns(false); - } + private void MockFileNotExists(string filePath) => fileSystem.File.Exists(filePath).Returns(false); } diff --git a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs index 6de8ea999..72d5b7ce8 100644 --- a/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs +++ b/src/ConnectedMode.UnitTests/Persistence/SolutionBindingRepositoryTests.cs @@ -30,21 +30,21 @@ namespace SonarLint.VisualStudio.ConnectedMode.UnitTests.Persistence; [TestClass] public class SolutionBindingRepositoryTests { - private IUnintrusiveBindingPathProvider unintrusiveBindingPathProvider; + private const string MockFilePath = "test file path"; + private const string LocalBindingKey = "my solution"; + + private BindingJsonModel bindingJsonModel; private IBindingJsonModelConverter bindingJsonModelConverter; - private IServerConnectionsRepository serverConnectionsRepository; + private BoundServerProject boundServerProject; private ISolutionBindingCredentialsLoader credentialsLoader; - private ISolutionBindingFileLoader solutionBindingFileLoader; private TestLogger logger; - private BindingJsonModel bindingJsonModel; + private BasicAuthCredentials mockCredentials; private ServerConnection serverConnection; - private BoundServerProject boundServerProject; + private IServerConnectionsRepository serverConnectionsRepository; + private ISolutionBindingFileLoader solutionBindingFileLoader; private ISolutionBindingRepository testSubject; - - private BasicAuthCredentials mockCredentials; - private const string MockFilePath = "test file path"; - private const string LocalBindingKey = "my solution"; + private IUnintrusiveBindingPathProvider unintrusiveBindingPathProvider; [TestInitialize] public void TestInitialize() @@ -62,10 +62,7 @@ public void TestInitialize() serverConnection = new ServerConnection.SonarCloud("org"); boundServerProject = new BoundServerProject("solution.123", "project_123", serverConnection); - bindingJsonModel = new BindingJsonModel - { - ServerConnectionId = serverConnection.Id - }; + bindingJsonModel = new BindingJsonModel { ServerConnectionId = serverConnection.Id }; } [TestMethod] @@ -86,10 +83,7 @@ public void MefCtor_CheckIsExported() } [TestMethod] - public void MefCtor_CheckIsSingleton() - { - MefTestHelpers.CheckIsSingletonMefComponent(); - } + public void MefCtor_CheckIsSingleton() => MefTestHelpers.CheckIsSingletonMefComponent(); [TestMethod] public void Read_ProjectIsNull_Null() @@ -125,10 +119,10 @@ public void Read_ProjectIsNotNull_ReadsConnectionRepositoryForConnection() var actual = testSubject.Read(MockFilePath); actual.Should().BeSameAs(boundServerProject); - + credentialsLoader.DidNotReceiveWithAnyArgs().Load(default); } - + [TestMethod] public void Read_ProjectIsNotNull_NoConnection_ReturnsNull() { @@ -140,7 +134,7 @@ public void Read_ProjectIsNotNull_NoConnection_ReturnsNull() solutionBindingFileLoader.Load(MockFilePath).Returns(bindingJsonModel); var actual = testSubject.Read(MockFilePath); - + credentialsLoader.DidNotReceiveWithAnyArgs().Load(default); actual.Should().BeNull(); } @@ -153,31 +147,28 @@ public void Write_ConfigFilePathIsNull_ReturnsFalse(string filePath) var actual = testSubject.Write(filePath, boundServerProject); actual.Should().Be(false); } - + [DataTestMethod] [DataRow(null)] [DataRow("")] public void Write_ConfigFilePathIsNull_FileNotWritten(string filePath) { testSubject.Write(filePath, boundServerProject); - + solutionBindingFileLoader.DidNotReceiveWithAnyArgs().Save(default, default); } - + [TestMethod] - public void Write_ProjectIsNull_Exception() - { - Assert.ThrowsException(() => testSubject.Write(MockFilePath, null)); - } - + public void Write_ProjectIsNull_Exception() => Assert.ThrowsException(() => testSubject.Write(MockFilePath, null)); + [TestMethod] public void Write_ProjectIsNull_FileNotWritten() { Assert.ThrowsException(() => testSubject.Write(MockFilePath, null)); - + solutionBindingFileLoader.DidNotReceiveWithAnyArgs().Save(default, default); } - + [DataTestMethod] [DataRow(true)] [DataRow(false)] @@ -187,23 +178,22 @@ public void Write_EventTriggered_DependingOnFileWriteStatus(bool triggered) testSubject.BindingUpdated += eventHandler; bindingJsonModelConverter.ConvertToModel(boundServerProject).Returns(bindingJsonModel); solutionBindingFileLoader.Save(MockFilePath, bindingJsonModel).Returns(triggered); - + testSubject.Write(MockFilePath, boundServerProject); - + eventHandler.ReceivedWithAnyArgs(triggered ? 1 : 0).Invoke(default, default); } - - + [TestMethod] public void Write_FileWritten_NoOnSaveCallback_NoException() { bindingJsonModelConverter.ConvertToModel(boundServerProject).Returns(bindingJsonModel); solutionBindingFileLoader.Save(MockFilePath, bindingJsonModel).Returns(true); - + Action act = () => testSubject.Write(MockFilePath, boundServerProject); act.Should().NotThrow(); } - + [TestMethod] public void List_FilesExist_Returns() { @@ -217,9 +207,9 @@ public void List_FilesExist_Returns() SetUpConnections(connection1, connection2); var boundServerProject1 = SetUpBinding(solution1, connection1, bindingConfig1); var boundServerProject2 = SetUpBinding(solution2, connection2, bindingConfig2); - + var result = testSubject.List(); - + result.Should().BeEquivalentTo(boundServerProject1, boundServerProject2); } @@ -236,12 +226,12 @@ public void List_SkipsBindingsWithoutConnections() SetUpConnections(connection2); // only one connection _ = SetUpBinding(solution1, null, bindingConfig1); var boundServerProject2 = SetUpBinding(solution2, connection2, bindingConfig2); - + var result = testSubject.List(); result.Should().BeEquivalentTo(boundServerProject2); } - + [TestMethod] public void List_SkipsBindingsThatCannotBeRead() { @@ -254,17 +244,17 @@ public void List_SkipsBindingsThatCannotBeRead() SetUpConnections(connection1, connection2); var boundServerProject1 = SetUpBinding(solution1, connection1, bindingConfig1); solutionBindingFileLoader.Load(bindingConfig2).Returns((BindingJsonModel)null); - + var result = testSubject.List(); result.Should().BeEquivalentTo(boundServerProject1); } - + [TestMethod] public void List_CannotGetConnections_EmptyList() { serverConnectionsRepository.TryGetAll(out Arg.Any>()).Returns(false); - + var act = () => testSubject.List().ToList(); act.Should().Throw(); @@ -274,11 +264,11 @@ public void List_CannotGetConnections_EmptyList() public void LegacyRead_NoFile_ReturnsNull() { solutionBindingFileLoader.Load(MockFilePath).Returns((BindingJsonModel)null); - + ((ILegacySolutionBindingRepository)testSubject).Read(MockFilePath).Should().BeNull(); credentialsLoader.DidNotReceiveWithAnyArgs().Load(default); } - + [TestMethod] public void LegacyRead_ValidBinding_LoadsCredentials() { @@ -287,7 +277,7 @@ public void LegacyRead_ValidBinding_LoadsCredentials() credentialsLoader.Load(bindingJsonModel.ServerUri).Returns(mockCredentials); solutionBindingFileLoader.Load(MockFilePath).Returns(bindingJsonModel); bindingJsonModelConverter.ConvertFromModelToLegacy(bindingJsonModel, mockCredentials).Returns(boundSonarQubeProject); - + ((ILegacySolutionBindingRepository)testSubject).Read(MockFilePath).Should().BeSameAs(boundSonarQubeProject); credentialsLoader.Received().Load(bindingJsonModel.ServerUri); } @@ -317,7 +307,7 @@ public void DeleteBinding_ReturnsResultOfDeleteBindingDirectory(bool expectedRes private BoundServerProject SetUpBinding(string solution, ServerConnection connection, string bindingConfig) { - var dto = new BindingJsonModel{ServerConnectionId = connection?.Id}; + var dto = new BindingJsonModel { ServerConnectionId = connection?.Id }; solutionBindingFileLoader.Load(bindingConfig).Returns(dto); if (connection == null) { @@ -328,9 +318,8 @@ private BoundServerProject SetUpBinding(string solution, ServerConnection connec bindingJsonModelConverter.ConvertFromModel(dto, connection, solution).Returns(bound); return bound; } - - private void SetUpConnections(params ServerConnection[] connections) - { + + private void SetUpConnections(params ServerConnection[] connections) => serverConnectionsRepository .TryGetAll(out Arg.Any>()) .Returns(call => @@ -338,10 +327,6 @@ private void SetUpConnections(params ServerConnection[] connections) call[0] = connections; return true; }); - } - private void SetUpUnintrusiveBindingPathProvider(params string[] bindigFolders) - { - unintrusiveBindingPathProvider.GetBindingPaths().Returns(bindigFolders); - } + private void SetUpUnintrusiveBindingPathProvider(params string[] bindigFolders) => unintrusiveBindingPathProvider.GetBindingPaths().Returns(bindigFolders); } diff --git a/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs b/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs index 63224a026..d1c49cd8e 100644 --- a/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs +++ b/src/ConnectedMode/Persistence/ISolutionBindingFileLoader.cs @@ -18,12 +18,13 @@ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ -namespace SonarLint.VisualStudio.ConnectedMode.Persistence +namespace SonarLint.VisualStudio.ConnectedMode.Persistence; + +internal interface ISolutionBindingFileLoader { - internal interface ISolutionBindingFileLoader - { - BindingJsonModel Load(string filePath); - bool Save(string filePath, BindingJsonModel project); - bool DeleteBindingDirectory(string configFilePath); - } + BindingJsonModel Load(string filePath); + + bool Save(string filePath, BindingJsonModel project); + + bool DeleteBindingDirectory(string configFilePath); } diff --git a/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs b/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs index 2ec1c1478..6bf1b6ee0 100644 --- a/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs +++ b/src/ConnectedMode/Persistence/SolutionBindingFileLoader.cs @@ -18,129 +18,114 @@ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ -using System; -using System.Diagnostics; using System.IO; using System.IO.Abstractions; using Newtonsoft.Json; using SonarLint.VisualStudio.Core; -using SonarLint.VisualStudio.Core.Binding; +using ErrorHandler = Microsoft.VisualStudio.ErrorHandler; -namespace SonarLint.VisualStudio.ConnectedMode.Persistence +namespace SonarLint.VisualStudio.ConnectedMode.Persistence; + +internal class SolutionBindingFileLoader : ISolutionBindingFileLoader { - internal class SolutionBindingFileLoader : ISolutionBindingFileLoader - { - private readonly ILogger logger; - private readonly IFileSystem fileSystem; + private readonly IFileSystem fileSystem; + private readonly ILogger logger; - public SolutionBindingFileLoader(ILogger logger) - : this(logger, new FileSystem()) - { - } + public SolutionBindingFileLoader(ILogger logger) + : this(logger, new FileSystem()) + { + } - internal SolutionBindingFileLoader(ILogger logger, IFileSystem fileSystem) - { - this.logger = logger ?? throw new ArgumentNullException(nameof(logger)); - this.fileSystem = fileSystem ?? throw new ArgumentNullException(nameof(fileSystem)); - } + internal SolutionBindingFileLoader(ILogger logger, IFileSystem fileSystem) + { + this.logger = logger ?? throw new ArgumentNullException(nameof(logger)); + this.fileSystem = fileSystem ?? throw new ArgumentNullException(nameof(fileSystem)); + } - public bool Save(string filePath, BindingJsonModel project) - { - var serializedProject = Serialize(project); + public bool Save(string filePath, BindingJsonModel project) + { + var serializedProject = Serialize(project); - return SafePerformFileSystemOperation(() => WriteConfig(filePath, serializedProject)); - } + return SafePerformFileSystemOperation(() => WriteConfig(filePath, serializedProject)); + } - public bool DeleteBindingDirectory(string configFilePath) + public bool DeleteBindingDirectory(string configFilePath) + { + if (fileSystem.File.Exists(configFilePath)) { - if (fileSystem.File.Exists(configFilePath)) + return SafePerformFileSystemOperation(() => { - return SafePerformFileSystemOperation(() => - { - var directoryName = Path.GetDirectoryName(configFilePath); - fileSystem.Directory.Delete(directoryName, recursive:true); - }); - } - - logger.LogVerbose(PersistenceStrings.BindingDirectoryNotDeleted, configFilePath); - return false; + var directoryName = Path.GetDirectoryName(configFilePath); + fileSystem.Directory.Delete(directoryName, true); + }); } - private void WriteConfig(string configFile, string serializedProject) + logger.LogVerbose(PersistenceStrings.BindingDirectoryNotDeleted, configFilePath); + return false; + } + + public BindingJsonModel Load(string filePath) + { + if (string.IsNullOrEmpty(filePath) || !fileSystem.File.Exists(filePath)) { - Debug.Assert(!string.IsNullOrWhiteSpace(configFile)); + return null; + } - var directoryName = Path.GetDirectoryName(configFile); + string configJson = null; - if (!fileSystem.Directory.Exists(directoryName)) + if (SafePerformFileSystemOperation(() => ReadConfig(filePath, out configJson))) + { + try { - fileSystem.Directory.CreateDirectory(directoryName); + return Deserialize(configJson); } - - fileSystem.File.WriteAllText(configFile, serializedProject); - } - - public BindingJsonModel Load(string filePath) - { - if (string.IsNullOrEmpty(filePath) || !fileSystem.File.Exists(filePath)) + catch (JsonException) { - return null; + logger.WriteLine(PersistenceStrings.FailedToDeserializeSQCOnfiguration, filePath); } + } - string configJson = null; + return null; + } - if (SafePerformFileSystemOperation(() => ReadConfig(filePath, out configJson))) - { - try - { - return Deserialize(configJson); - } - catch (JsonException) - { - logger.WriteLine(PersistenceStrings.FailedToDeserializeSQCOnfiguration, filePath); - } - } + private void WriteConfig(string configFile, string serializedProject) + { + Debug.Assert(!string.IsNullOrWhiteSpace(configFile)); - return null; - } + var directoryName = Path.GetDirectoryName(configFile); - private void ReadConfig(string configFile, out string text) + if (!fileSystem.Directory.Exists(directoryName)) { - text = fileSystem.File.ReadAllText(configFile); + fileSystem.Directory.CreateDirectory(directoryName); } - private bool SafePerformFileSystemOperation(Action operation) - { - Debug.Assert(operation != null); + fileSystem.File.WriteAllText(configFile, serializedProject); + } - try - { - operation(); - return true; - } - catch (Exception e) when (!Microsoft.VisualStudio.ErrorHandler.IsCriticalException(e)) - { - logger.WriteLine(e.Message); - return false; - } - } + private void ReadConfig(string configFile, out string text) => text = fileSystem.File.ReadAllText(configFile); + + private bool SafePerformFileSystemOperation(Action operation) + { + Debug.Assert(operation != null); - private BindingJsonModel Deserialize(string projectJson) + try { - return JsonConvert.DeserializeObject(projectJson, new JsonSerializerSettings - { - DateFormatHandling = DateFormatHandling.IsoDateFormat, - DateTimeZoneHandling = DateTimeZoneHandling.Local, - DateParseHandling = DateParseHandling.DateTimeOffset - }); + operation(); + return true; } - - private string Serialize(BindingJsonModel project) + catch (Exception e) when (!ErrorHandler.IsCriticalException(e)) { - return JsonConvert.SerializeObject(project, Formatting.Indented, new JsonSerializerSettings - { - DateTimeZoneHandling = DateTimeZoneHandling.Utc - }); + logger.WriteLine(e.Message); + return false; } } + + private BindingJsonModel Deserialize(string projectJson) => + JsonConvert.DeserializeObject(projectJson, + new JsonSerializerSettings + { + DateFormatHandling = DateFormatHandling.IsoDateFormat, DateTimeZoneHandling = DateTimeZoneHandling.Local, DateParseHandling = DateParseHandling.DateTimeOffset + }); + + private string Serialize(BindingJsonModel project) => JsonConvert.SerializeObject(project, Formatting.Indented, new JsonSerializerSettings { DateTimeZoneHandling = DateTimeZoneHandling.Utc }); } diff --git a/src/Core/Binding/ISolutionBindingRepository.cs b/src/Core/Binding/ISolutionBindingRepository.cs index ca63684c6..e0909422b 100644 --- a/src/Core/Binding/ISolutionBindingRepository.cs +++ b/src/Core/Binding/ISolutionBindingRepository.cs @@ -46,7 +46,7 @@ public interface ISolutionBindingRepository /// /// Deletes the binding information /// - /// The local binding key of the + /// The local binding key of the /// If binding has been deleted bool DeleteBinding(string localBindingKey); From 1a8f2a98fc73982c46baf9912fcf07ef3d4f9971 Mon Sep 17 00:00:00 2001 From: Gabriela Trutan Date: Thu, 5 Dec 2024 11:03:37 +0100 Subject: [PATCH 5/5] SLVS-1404 Apply review feedback --- src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs | 2 +- src/ConnectedMode/Persistence/PersistenceStrings.resx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs b/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs index b1ebadff7..b1151b7a4 100644 --- a/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs +++ b/src/ConnectedMode/Persistence/PersistenceStrings.Designer.cs @@ -61,7 +61,7 @@ internal PersistenceStrings() { } /// - /// Looks up a localized string similar to The binding configuration file '{0}' does not exist. Binding directory not deleted.. + /// Looks up a localized string similar to The Connected Mode configuration file '{0}' does not exist. Connected Mode configuration not deleted.. /// internal static string BindingDirectoryNotDeleted { get { diff --git a/src/ConnectedMode/Persistence/PersistenceStrings.resx b/src/ConnectedMode/Persistence/PersistenceStrings.resx index 4758ee42a..871b02324 100644 --- a/src/ConnectedMode/Persistence/PersistenceStrings.resx +++ b/src/ConnectedMode/Persistence/PersistenceStrings.resx @@ -122,6 +122,6 @@ Output window message in case of deserialization error. {0} file path. - The binding configuration file '{0}' does not exist. Binding directory not deleted. + The Connected Mode configuration file '{0}' does not exist. Connected Mode configuration not deleted. \ No newline at end of file