From 0f1a4b9b2672ed7e5b9a4bede46af69febdd696f Mon Sep 17 00:00:00 2001 From: Georg Hinkel Date: Wed, 27 Nov 2024 14:39:03 +0100 Subject: [PATCH 1/7] simplify creating custom layout service --- .../Layouting/LayeredLayoutService.cs | 32 +++++++++++++++---- 1 file changed, 26 insertions(+), 6 deletions(-) diff --git a/Glsp/Glsp/Processing/Layouting/LayeredLayoutService.cs b/Glsp/Glsp/Processing/Layouting/LayeredLayoutService.cs index c96c1dde..8c37aa65 100644 --- a/Glsp/Glsp/Processing/Layouting/LayeredLayoutService.cs +++ b/Glsp/Glsp/Processing/Layouting/LayeredLayoutService.cs @@ -20,15 +20,35 @@ public class LayeredLayoutService : AglLayoutService /// public static readonly LayeredLayoutService Instance = new LayeredLayoutService(); + private static readonly SugiyamaLayoutSettings _defaultSettings = new SugiyamaLayoutSettings + { + Transformation = PlaneTransformation.Rotation(Math.PI / 2), + EdgeRoutingSettings = { EdgeRoutingMode = EdgeRoutingMode.Rectilinear } + }; + + /// + /// Creates a new instance + /// + public LayeredLayoutService() : this(null) { } + + /// + /// Creates a new instance + /// + /// layout settings + public LayeredLayoutService(SugiyamaLayoutSettings settings) + { + Settings = settings ?? _defaultSettings; + } + + /// + /// Gets the layout settings for this layout service + /// + public SugiyamaLayoutSettings Settings { get; } + /// protected override void ProcessLayout(GeometryGraph g) { - var settings = new SugiyamaLayoutSettings - { - Transformation = PlaneTransformation.Rotation(Math.PI / 2), - EdgeRoutingSettings = { EdgeRoutingMode = EdgeRoutingMode.Rectilinear } - }; - var layout = new LayeredLayout(g, settings); + var layout = new LayeredLayout(g, Settings); layout.Run(); } } From 81b8352193e8fd9a3560615a6324305da928327a Mon Sep 17 00:00:00 2001 From: Georg Hinkel Date: Tue, 10 Dec 2024 08:26:40 +0100 Subject: [PATCH 2/7] create an explicit type constructor for MetaRepository --- Models/Models/Repository/MetaRepository.cs | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/Models/Models/Repository/MetaRepository.cs b/Models/Models/Repository/MetaRepository.cs index 2ca17999..fce4314d 100644 --- a/Models/Models/Repository/MetaRepository.cs +++ b/Models/Models/Repository/MetaRepository.cs @@ -13,11 +13,21 @@ namespace NMF.Models.Repository /// public sealed class MetaRepository : IModelRepository { - private static readonly MetaRepository instance = new MetaRepository(); + private static readonly MetaRepository instance; private readonly ModelCollection entries; private readonly ModelSerializer serializer = new ModelSerializer(); private readonly HashSet traversedAssemblies = new HashSet(); + /// + /// Initializes the type + /// + static MetaRepository() + { + // we need an explicit static type constructor because the runtime is sometimes too lazy + // to initialize static variables in time, but they are needed for static lookup operations + instance = new MetaRepository(); + } + event EventHandler IModelRepository.BubbledChange { add { } From 1e40c9647fe0ef3828c5416deb488d8e4460f7bf Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 9 Oct 2024 01:13:25 +0000 Subject: [PATCH 3/7] Bump System.Text.Json from 8.0.4 to 8.0.5 in /Models/Serialization.Json Bumps [System.Text.Json](https://github.com/dotnet/runtime) from 8.0.4 to 8.0.5. - [Release notes](https://github.com/dotnet/runtime/releases) - [Commits](https://github.com/dotnet/runtime/compare/v8.0.4...v8.0.5) --- updated-dependencies: - dependency-name: System.Text.Json dependency-type: direct:production ... Signed-off-by: dependabot[bot] --- Models/Serialization.Json/Serialization.Json.csproj | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Models/Serialization.Json/Serialization.Json.csproj b/Models/Serialization.Json/Serialization.Json.csproj index dca378cd..0f72d64a 100644 --- a/Models/Serialization.Json/Serialization.Json.csproj +++ b/Models/Serialization.Json/Serialization.Json.csproj @@ -26,7 +26,7 @@ - + From 33f82c7471480b174aab4231ab99839cf482530d Mon Sep 17 00:00:00 2001 From: ahert001 Date: Wed, 18 Dec 2024 20:09:53 +0100 Subject: [PATCH 4/7] Add Logs --- AnyText/AnyText.Lsp/LspServer.Diagnostics.cs | 6 ++- AnyText/AnyText.Lsp/LspServer.Registration.cs | 6 ++- .../AnyText.Lsp/LspServer.SemanticToken.cs | 3 ++ AnyText/AnyText.Lsp/LspServer.cs | 37 ++++++++++++++++--- 4 files changed, 45 insertions(+), 7 deletions(-) diff --git a/AnyText/AnyText.Lsp/LspServer.Diagnostics.cs b/AnyText/AnyText.Lsp/LspServer.Diagnostics.cs index 1afcce79..e05e2878 100644 --- a/AnyText/AnyText.Lsp/LspServer.Diagnostics.cs +++ b/AnyText/AnyText.Lsp/LspServer.Diagnostics.cs @@ -10,6 +10,7 @@ public partial class LspServer private async void SendDiagnostics(string uri, ParseContext context) { + SendLogMessage(MessageType.Info, $"Starting diagnostics generation for URI: {uri}"); var diagnostics = new List(); var errors = context.Errors; foreach (var error in errors) @@ -38,10 +39,13 @@ private async void SendDiagnostics(string uri, ParseContext context) try { await _rpc.NotifyWithParameterObjectAsync(Methods.TextDocumentPublishDiagnosticsName, diagnosticsParams); + SendLogMessage(MessageType.Info, $"Diagnostics published successfully for URI: {uri} with {diagnostics.Count} issue(s)."); } catch (Exception ex) { - Console.Error.WriteLine($"Error publishing Diagnostics: {ex.Message}"); + var errorMessage = $"Error publishing diagnostics for URI: {uri}. Exception: {ex.Message}"; + SendLogMessage(MessageType.Error, errorMessage); + Console.Error.WriteLine(errorMessage); } } } diff --git a/AnyText/AnyText.Lsp/LspServer.Registration.cs b/AnyText/AnyText.Lsp/LspServer.Registration.cs index ddaed1b7..a005b221 100644 --- a/AnyText/AnyText.Lsp/LspServer.Registration.cs +++ b/AnyText/AnyText.Lsp/LspServer.Registration.cs @@ -35,11 +35,15 @@ private async void RegisterCapabilities(Registration[] registrations) try { + SendLogMessage(MessageType.Info, "Sending capabilities registration request to client."); await _rpc.InvokeWithParameterObjectAsync(Methods.ClientRegisterCapabilityName, registrationParams); + SendLogMessage(MessageType.Info, "Capabilities registration request completed successfully."); } catch (Exception ex) { - Console.Error.WriteLine($"Error register capabilities: {ex.Message}"); + var errorMessage = $"Error registering capabilities: {ex.Message}"; + SendLogMessage(MessageType.Error, errorMessage); + Console.Error.WriteLine(errorMessage); } } diff --git a/AnyText/AnyText.Lsp/LspServer.SemanticToken.cs b/AnyText/AnyText.Lsp/LspServer.SemanticToken.cs index d103a14e..4697dd9f 100644 --- a/AnyText/AnyText.Lsp/LspServer.SemanticToken.cs +++ b/AnyText/AnyText.Lsp/LspServer.SemanticToken.cs @@ -13,6 +13,7 @@ public partial class LspServer /// public SemanticTokens QuerySemanticTokens(JToken arg) { + SendLogMessage(MessageType.Info, "Received request for full semantic tokens."); var semanticTokensParams = arg.ToObject(); var uri = semanticTokensParams.TextDocument.Uri; @@ -33,6 +34,7 @@ public SemanticTokens QuerySemanticTokens(JToken arg) /// public SemanticTokensDelta QuerySemanticTokensDelta(JToken arg) { + SendLogMessage(MessageType.Info, "Received request for semantic tokens delta."); var semanticTokensParams = arg.ToObject(); var uri = semanticTokensParams.TextDocument.Uri; @@ -74,6 +76,7 @@ public SemanticTokensDelta QuerySemanticTokensDelta(JToken arg) /// public SemanticTokens QuerySemanticTokensRange(JToken arg) { + SendLogMessage(MessageType.Info, "Received request for semantic tokens in a range."); var semanticTokensRangeParams = arg.ToObject(); var uri = semanticTokensRangeParams.TextDocument.Uri; var range = semanticTokensRangeParams.Range; diff --git a/AnyText/AnyText.Lsp/LspServer.cs b/AnyText/AnyText.Lsp/LspServer.cs index 57f917f3..12a77e65 100644 --- a/AnyText/AnyText.Lsp/LspServer.cs +++ b/AnyText/AnyText.Lsp/LspServer.cs @@ -1,4 +1,4 @@ -using LspTypes; +using LspTypes; using Newtonsoft.Json; using Newtonsoft.Json.Linq; using NMF.AnyText.Grammars; @@ -39,7 +39,6 @@ public LspServer(IEnumerable grammars) _languages = grammars?.ToDictionary(sp => sp.LanguageId); } - [JsonRpcMethod(Methods.InitializeName)] public InitializeResult Initialize( int? processId , _InitializeParams_ClientInfo clientInfo @@ -68,8 +67,10 @@ public InitializeResult Initialize( WorkDoneProgress = false } }; + SendLogMessage(MessageType.Info, "LSP Server initialization completed."); return new InitializeResult { Capabilities = serverCapabilities }; } + public void Initialized() { } public void Shutdown() { } @@ -83,11 +84,13 @@ public void DidChange(JToken arg) { document.Update(changes.ContentChanges.Select(AsTextEdit)); SendDiagnostics(changes.TextDocument.Uri, document.Context); + SendLogMessage(MessageType.Info, $"Document {changes.TextDocument.Uri} updated."); } } public void DidSave(TextDocumentIdentifier textDocument, string text) { + SendLogMessage(MessageType.Info, $"Document {textDocument.Uri} saved."); } private static ParsePosition AsParsePosition(Position position) => new ParsePosition((int)position.Line, (int)position.Character); @@ -104,6 +107,7 @@ public void DidClose(JToken arg) if (_documents.TryGetValue(closeParams.TextDocument.Uri, out var document)) { _documents.Remove(closeParams.TextDocument.Uri); + SendLogMessage(MessageType.Info, $"Document {closeParams.TextDocument.Uri} closed."); } } @@ -117,25 +121,48 @@ public void DidOpen(JToken arg) if (_languages.TryGetValue(openParams.TextDocument.LanguageId, out var language)) { var parser = language.CreateParser(); - parser.Initialize(File.ReadAllLines(uri.AbsolutePath)); + parser.Initialize(File.ReadAllLines(uri.AbsolutePath).Concat(new[] { string.Empty }).ToArray()); _documents.Add(openParams.TextDocument.Uri, parser); RegisterCapabilitiesOnOpen(openParams.TextDocument.LanguageId, parser); SendDiagnostics(openParams.TextDocument.Uri, parser.Context); + SendLogMessage(MessageType.Info, $"Document {openParams.TextDocument.Uri} opened with language {openParams.TextDocument.LanguageId}."); } else { - throw new NotSupportedException($"No grammar found for extension {openParams.TextDocument.LanguageId}"); + var errorMessage = $"No grammar found for extension {openParams.TextDocument.LanguageId}"; + SendLogMessage(MessageType.Error, errorMessage); + throw new NotSupportedException(errorMessage); } } else { - throw new NotSupportedException($"Cannot open URI {openParams.TextDocument.Uri}"); + var errorMessage = $"Cannot open URI {openParams.TextDocument.Uri}"; + SendLogMessage(MessageType.Error, errorMessage); + throw new NotSupportedException(errorMessage); } } public void Exit() { + SendLogMessage(MessageType.Info, "LSP Server exiting."); throw new NotImplementedException(); } + + // + /// Sends a log message to the client. + /// + /// The type of the message (Info, Warning, Error). + /// The message content. + private void SendLogMessage(MessageType type, string message) + { + var logMessageParams = new LogMessageParams + { + MessageType = type, + Message = message + }; + + _rpc.NotifyWithParameterObjectAsync(Methods.WindowLogMessageName, logMessageParams); + + } } } \ No newline at end of file From e90070dd98fd3973196b90e3166dd99027ada3b3 Mon Sep 17 00:00:00 2001 From: ahert001 Date: Thu, 19 Dec 2024 08:46:10 +0100 Subject: [PATCH 5/7] add setTrace RPC-Method --- .../AnyText.Lsp/AnyTextJsonRpcServerUtil.cs | 21 ++++++------ AnyText/AnyText.Lsp/ILspServer.cs | 13 ++++--- AnyText/AnyText.Lsp/LspServer.Trace.cs | 34 +++++++++++++++++++ AnyText/AnyText.Lsp/LspServer.cs | 2 ++ AnyText/AnyText.Lsp/MethodConstants.cs | 1 + AnyText/Tests/AnyTextExtension/package.json | 4 +-- 6 files changed, 58 insertions(+), 17 deletions(-) create mode 100644 AnyText/AnyText.Lsp/LspServer.Trace.cs diff --git a/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs b/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs index c4ad0986..226b7f6d 100644 --- a/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs +++ b/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs @@ -11,8 +11,6 @@ namespace NMF.AnyText /// public static class AnyTextJsonRpcServerUtil { - private static readonly TraceSource _traceSource = CreateTraceSource(); - /// /// Creates a StreamJSON RPC object for the given transport /// @@ -22,7 +20,6 @@ public static class AnyTextJsonRpcServerUtil public static JsonRpc CreateServer(WebSocket webSocket, ILspServer server) { var rpc = new JsonRpc(new WebSocketMessageHandler(webSocket, CreateFormatter())); - rpc.TraceSource = _traceSource; rpc.AddLocalRpcTarget(server, CreateTargetOptions()); return rpc; } @@ -36,7 +33,6 @@ public static JsonRpc CreateServer(WebSocket webSocket, ILspServer server) public static JsonRpc CreateServer(IDuplexPipe pipe, ILspServer server) { var rpc = new JsonRpc(new HeaderDelimitedMessageHandler(pipe, CreateFormatter())); - rpc.TraceSource = _traceSource; rpc.AddLocalRpcTarget(server, CreateTargetOptions()); return rpc; } @@ -50,7 +46,6 @@ public static JsonRpc CreateServer(IDuplexPipe pipe, ILspServer server) public static JsonRpc CreateServer(Stream stream, ILspServer server) { var rpc = new JsonRpc(new HeaderDelimitedMessageHandler(stream, CreateFormatter())); - rpc.TraceSource = _traceSource; rpc.AddLocalRpcTarget(server, CreateTargetOptions()); return rpc; } @@ -62,7 +57,6 @@ public static JsonRpc CreateServer(Stream stream, ILspServer server) public static JsonRpc CreateServer(Stream stream) { var rpc = new JsonRpc(new HeaderDelimitedMessageHandler(stream, CreateFormatter())); - rpc.TraceSource = _traceSource; return rpc; } @@ -84,15 +78,20 @@ private static JsonRpcTargetOptions CreateTargetOptions() EventNameTransform = name => name.ToLowerInvariant() }; } - - private static TraceSource CreateTraceSource() + /// + /// Creates and configures a instance for logging trace information. + /// + /// The SourceLevel used to filter messages by type and severity. Defaults to . + /// A instance configured for the specified logging level. + public static TraceSource CreateTraceSource(SourceLevels sourceLevels = SourceLevels.All) { - var traceSource = new TraceSource("LSP", SourceLevels.All); - // use error stream such that VS Code can see the stdout - traceSource.Listeners.Add(new ConsoleTraceListener(true)); + var traceSource = new TraceSource("LSP", sourceLevels); + // Use error stream (stderr) so that VS Code can capture the output + //traceSource.Listeners.Add(new ConsoleTraceListener(useErrorStream: true)); return traceSource; } + private static IJsonRpcMessageFormatter CreateFormatter() { return new JsonMessageFormatter(); diff --git a/AnyText/AnyText.Lsp/ILspServer.cs b/AnyText/AnyText.Lsp/ILspServer.cs index 062fea25..43a0bc0c 100644 --- a/AnyText/AnyText.Lsp/ILspServer.cs +++ b/AnyText/AnyText.Lsp/ILspServer.cs @@ -37,11 +37,16 @@ public InitializeResult Initialize( [JsonRpcMethod(Methods.InitializedName)] void Initialized(); - - - - + [JsonRpcMethod(Methods.ShutdownName)] void Shutdown(); + + /// + /// Handles the */setTrace request from the client. This is used to set the trace setting of the server. + /// + /// The JSON token containing the parameters of the request. (SetTraceParams) + [JsonRpcMethod(MethodConstants.SetTrace)] + public void SetTrace(JToken arg); + } } \ No newline at end of file diff --git a/AnyText/AnyText.Lsp/LspServer.Trace.cs b/AnyText/AnyText.Lsp/LspServer.Trace.cs new file mode 100644 index 00000000..61a33a2d --- /dev/null +++ b/AnyText/AnyText.Lsp/LspServer.Trace.cs @@ -0,0 +1,34 @@ +using LspTypes; +using Newtonsoft.Json.Linq; +using System.Diagnostics; + +namespace NMF.AnyText +{ + public partial class LspServer + { + /// + public void SetTrace(JToken arg) + { + var setTraceParams = arg.ToObject(); + + UpdateTraceSource(setTraceParams.Value); + SendLogMessage(MessageType.Info, $"Trace level updated to: {setTraceParams.Value}"); + } + + private void UpdateTraceSource(TraceValue traceValue) + { + _rpc.TraceSource = + AnyTextJsonRpcServerUtil.CreateTraceSource(MapTraceValueToSourceLevels(traceValue)); + } + private static SourceLevels MapTraceValueToSourceLevels(TraceValue traceValue) + { + return traceValue switch + { + TraceValue.Off => SourceLevels.Off, + TraceValue.Messages => SourceLevels.Information, + TraceValue.Verbose => SourceLevels.Verbose, + _ => SourceLevels.All + }; + } + } +} \ No newline at end of file diff --git a/AnyText/AnyText.Lsp/LspServer.cs b/AnyText/AnyText.Lsp/LspServer.cs index 12a77e65..3a61aa24 100644 --- a/AnyText/AnyText.Lsp/LspServer.cs +++ b/AnyText/AnyText.Lsp/LspServer.cs @@ -67,6 +67,8 @@ public InitializeResult Initialize( WorkDoneProgress = false } }; + UpdateTraceSource(trace); + SendLogMessage(MessageType.Info, "LSP Server initialization completed."); return new InitializeResult { Capabilities = serverCapabilities }; } diff --git a/AnyText/AnyText.Lsp/MethodConstants.cs b/AnyText/AnyText.Lsp/MethodConstants.cs index 2327fe9d..3054d8d1 100644 --- a/AnyText/AnyText.Lsp/MethodConstants.cs +++ b/AnyText/AnyText.Lsp/MethodConstants.cs @@ -3,5 +3,6 @@ namespace NMF.AnyText public static class MethodConstants { public const string RegisterSemanticTokens = "textDocument/semanticTokens"; + public const string SetTrace = "$/setTrace"; } } \ No newline at end of file diff --git a/AnyText/Tests/AnyTextExtension/package.json b/AnyText/Tests/AnyTextExtension/package.json index 14dee364..55fb94a7 100644 --- a/AnyText/Tests/AnyTextExtension/package.json +++ b/AnyText/Tests/AnyTextExtension/package.json @@ -38,13 +38,13 @@ "type": "object", "title": "Example configuration", "properties": { - "languageServerExample.maxNumberOfProblems": { + "anytext.maxNumberOfProblems": { "scope": "resource", "type": "number", "default": 100, "description": "Controls the maximum number of problems produced by the server." }, - "languageServerExample.trace.server": { + "anytext.trace.server": { "scope": "window", "type": "string", "enum": [ From 6a92e65958c0bb39d83f2d77263f614376d292eb Mon Sep 17 00:00:00 2001 From: ahert001 Date: Fri, 20 Dec 2024 11:37:39 +0100 Subject: [PATCH 6/7] enable Server Trace --- AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs b/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs index 226b7f6d..634b3313 100644 --- a/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs +++ b/AnyText/AnyText.Lsp/AnyTextJsonRpcServerUtil.cs @@ -87,7 +87,7 @@ public static TraceSource CreateTraceSource(SourceLevels sourceLevels = SourceLe { var traceSource = new TraceSource("LSP", sourceLevels); // Use error stream (stderr) so that VS Code can capture the output - //traceSource.Listeners.Add(new ConsoleTraceListener(useErrorStream: true)); + traceSource.Listeners.Add(new ConsoleTraceListener(useErrorStream: true)); return traceSource; } From 8a68f26643fafd0f8cd7896a14766f3789d6fc63 Mon Sep 17 00:00:00 2001 From: ahert001 Date: Fri, 20 Dec 2024 14:04:25 +0100 Subject: [PATCH 7/7] remove extra line onOpen --- AnyText/AnyText.Lsp/LspServer.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/AnyText/AnyText.Lsp/LspServer.cs b/AnyText/AnyText.Lsp/LspServer.cs index 3a61aa24..6e605314 100644 --- a/AnyText/AnyText.Lsp/LspServer.cs +++ b/AnyText/AnyText.Lsp/LspServer.cs @@ -123,7 +123,7 @@ public void DidOpen(JToken arg) if (_languages.TryGetValue(openParams.TextDocument.LanguageId, out var language)) { var parser = language.CreateParser(); - parser.Initialize(File.ReadAllLines(uri.AbsolutePath).Concat(new[] { string.Empty }).ToArray()); + parser.Initialize(File.ReadAllLines(uri.AbsolutePath)); _documents.Add(openParams.TextDocument.Uri, parser); RegisterCapabilitiesOnOpen(openParams.TextDocument.LanguageId, parser);