summaryrefslogtreecommitdiffstats
path: root/tests/dotnet/lsprotocol_tests
diff options
context:
space:
mode:
Diffstat (limited to 'tests/dotnet/lsprotocol_tests')
-rw-r--r--tests/dotnet/lsprotocol_tests/.gitignore2
-rw-r--r--tests/dotnet/lsprotocol_tests/LSPTests.cs123
-rw-r--r--tests/dotnet/lsprotocol_tests/Usings.cs2
-rw-r--r--tests/dotnet/lsprotocol_tests/lsprotocol_tests.csproj29
4 files changed, 156 insertions, 0 deletions
diff --git a/tests/dotnet/lsprotocol_tests/.gitignore b/tests/dotnet/lsprotocol_tests/.gitignore
new file mode 100644
index 0000000..cbbd0b5
--- /dev/null
+++ b/tests/dotnet/lsprotocol_tests/.gitignore
@@ -0,0 +1,2 @@
+bin/
+obj/ \ No newline at end of file
diff --git a/tests/dotnet/lsprotocol_tests/LSPTests.cs b/tests/dotnet/lsprotocol_tests/LSPTests.cs
new file mode 100644
index 0000000..80a19ae
--- /dev/null
+++ b/tests/dotnet/lsprotocol_tests/LSPTests.cs
@@ -0,0 +1,123 @@
+
+namespace lsprotocol_tests;
+using Newtonsoft.Json;
+using Newtonsoft.Json.Linq;
+
+
+public class LSPTests
+{
+ public static IEnumerable<object[]> JsonTestData()
+ {
+ string folderPath;
+ // Read test data path from environment variable
+ if (!string.IsNullOrEmpty(Environment.GetEnvironmentVariable("LSP_TEST_DATA_PATH")))
+ {
+ folderPath = Environment.GetEnvironmentVariable("LSP_TEST_DATA_PATH");
+ }
+ else
+ {
+ throw new Exception("LSP_TEST_DATA_PATH environment variable not set");
+ }
+
+ string[] jsonFiles = Directory.GetFiles(folderPath, "*.json");
+ foreach (string filePath in jsonFiles)
+ {
+ yield return new object[] { filePath };
+ }
+ }
+
+ [Theory]
+ [MemberData(nameof(JsonTestData))]
+ public void ValidateLSPTypes(string filePath)
+ {
+ string original = File.ReadAllText(filePath);
+
+ // Get the class name from the file name
+ // format: <class-name>-<valid>-<test-id>.json
+ // classname => Class name of the type to deserialize to
+ // valid => true if the file is valid, false if it is invalid
+ // test-id => unique id for the test
+ string fileName = Path.GetFileNameWithoutExtension(filePath);
+ string[] nameParts = fileName.Split('-');
+ string className = nameParts[0];
+ bool valid = nameParts[1] == "True";
+
+ Type type = Type.GetType($"Microsoft.LanguageServer.Protocol.{className}, lsprotocol") ?? throw new Exception($"Type {className} not found");
+ RunTest(valid, original, type);
+ }
+
+ private static void RunTest(bool valid, string data, Type type)
+ {
+ if (valid)
+ {
+ try
+ {
+ var settings = new JsonSerializerSettings
+ {
+ MissingMemberHandling = MissingMemberHandling.Error
+ };
+ object? deserializedObject = JsonConvert.DeserializeObject(data, type, settings);
+ string newJson = JsonConvert.SerializeObject(deserializedObject, settings);
+
+ JToken token1 = JToken.Parse(data);
+ JToken token2 = JToken.Parse(newJson);
+ RemoveNullProperties(token1);
+ RemoveNullProperties(token2);
+ Assert.True(JToken.DeepEquals(token1, token2), $"JSON before and after serialization don't match:\r\nBEFORE:{data}\r\nAFTER:{newJson}");
+ }
+ catch (Exception e)
+ {
+ // Explicitly fail the test
+ Assert.True(false, $"Should not have thrown an exception for [{type.Name}]: {data} \r\n{e}");
+ }
+ }
+ else
+ {
+ try
+ {
+ JsonConvert.DeserializeObject(data, type);
+ // Explicitly fail the test
+ Assert.True(false, $"Should have thrown an exception for [{type.Name}]: {data}");
+ }
+ catch
+ {
+ // Worked as expected.
+ }
+ }
+ }
+
+ private static void RemoveNullProperties(JToken token)
+ {
+ if (token.Type == JTokenType.Object)
+ {
+ var obj = (JObject)token;
+
+ var propertiesToRemove = obj.Properties()
+ .Where(p => p.Value.Type == JTokenType.Null)
+ .ToList();
+
+ foreach (var property in propertiesToRemove)
+ {
+ property.Remove();
+ }
+
+ foreach (var property in obj.Properties())
+ {
+ RemoveNullProperties(property.Value);
+ }
+ }
+ else if (token.Type == JTokenType.Array)
+ {
+ var array = (JArray)token;
+
+ for (int i = array.Count - 1; i >= 0; i--)
+ {
+ RemoveNullProperties(array[i]);
+ if (array[i].Type == JTokenType.Null)
+ {
+ array.RemoveAt(i);
+ }
+ }
+ }
+ }
+} \ No newline at end of file
diff --git a/tests/dotnet/lsprotocol_tests/Usings.cs b/tests/dotnet/lsprotocol_tests/Usings.cs
new file mode 100644
index 0000000..5d95411
--- /dev/null
+++ b/tests/dotnet/lsprotocol_tests/Usings.cs
@@ -0,0 +1,2 @@
+global using Xunit;
+global using Microsoft.LanguageServer.Protocol; \ No newline at end of file
diff --git a/tests/dotnet/lsprotocol_tests/lsprotocol_tests.csproj b/tests/dotnet/lsprotocol_tests/lsprotocol_tests.csproj
new file mode 100644
index 0000000..194c872
--- /dev/null
+++ b/tests/dotnet/lsprotocol_tests/lsprotocol_tests.csproj
@@ -0,0 +1,29 @@
+<Project Sdk="Microsoft.NET.Sdk">
+
+ <PropertyGroup>
+ <TargetFramework>net6.0</TargetFramework>
+ <ImplicitUsings>enable</ImplicitUsings>
+ <Nullable>enable</Nullable>
+
+ <IsPackable>false</IsPackable>
+ <IsTestProject>true</IsTestProject>
+ </PropertyGroup>
+
+ <ItemGroup>
+ <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.5.0" />
+ <PackageReference Include="xunit" Version="2.4.2" />
+ <PackageReference Include="xunit.runner.visualstudio" Version="2.4.5">
+ <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
+ <PrivateAssets>all</PrivateAssets>
+ </PackageReference>
+ <PackageReference Include="coverlet.collector" Version="3.2.0">
+ <IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
+ <PrivateAssets>all</PrivateAssets>
+ </PackageReference>
+ </ItemGroup>
+
+ <ItemGroup>
+ <ProjectReference Include="..\..\..\packages\dotnet\lsprotocol\lsprotocol.csproj" />
+ </ItemGroup>
+
+</Project>