This repository has been archived by the owner on Jan 12, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 169
/
Copy pathQsLanguageClient.cs
203 lines (174 loc) · 9.2 KB
/
QsLanguageClient.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License.
using Microsoft.VisualStudio.LanguageServer.Client;
using Microsoft.VisualStudio.Shell.Interop;
using Microsoft.VisualStudio.Threading;
using Microsoft.VisualStudio.Utilities;
using Newtonsoft.Json.Linq;
using StreamJsonRpc;
using System;
using System.Collections.Generic;
using System.ComponentModel.Composition;
using System.Diagnostics;
using System.IO;
using System.IO.Pipes;
using System.Reflection;
using System.Security.Principal;
using System.Text.RegularExpressions;
using System.Threading;
using System.Threading.Tasks;
using static System.Diagnostics.FileVersionInfo;
namespace Microsoft.Quantum.QsLanguageExtensionVS
{
[ContentType("Q#")]
[Export(typeof(ILanguageClient))]
public class QsLanguageClient : VisualStudio.Shell.AsyncPackage, ILanguageClient, ILanguageClientCustomMessage2
{
private readonly long LaunchTime;
public readonly string LogFile;
public readonly string LanguageServerPath;
public static Guid OutputPaneGuid =
new Guid("C6F36B68-90B4-4A12-BE58-34E3F735B0AE");
public QsLanguageClient() : base()
{
this.LaunchTime = DateTime.Now.Ticks;
this.LogFile = Path.Combine(Path.GetTempPath(), $"qsp-{LaunchTime}.log");
this.LanguageServerPath = Path.Combine(
Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location),
"LanguageServer", "Microsoft.Quantum.QsLanguageServer.exe");
CustomMessageTarget = new CustomServerNotifications();
}
// properties and methods required by ILanguageClientCustomMessage2
public object MiddleLayer => null; // we don't need to intercept messages
public object CustomMessageTarget { get; }
/// called third, before initializing the server
public Task AttachForCustomMessageAsync(JsonRpc rpc) =>
Task.CompletedTask; // we don't need to send custom messages
// properties required by ILanguageClient
public string Name => "Q# Language Extension"; // name as displayed to the user
public bool ShowNotificationOnInitializeFailed => true; // Will notify the user if Language Server failed to initialize
public IEnumerable<string> ConfigurationSections => null; // null is fine if the client does not provide settings
public IEnumerable<string> FilesToWatch => null; // we use our own watcher rather than the one of the LSP Client
public object InitializationOptions => JObject.FromObject(new
{
name = "VisualStudio",
version = GetVisualStudioVersion()
});
// events required by ILanguageClient
public event AsyncEventHandler<EventArgs> StartAsync;
#pragma warning disable 67 // The event StopAsync is never used, but is required by ILanguageClient.
public event AsyncEventHandler<EventArgs> StopAsync;
#pragma warning restore 67
// methods required by ILanguageClient
/// Called second, once the extension has been loaded.
/// Calls the StartAsync delegate to signal that the language server can/should be started.
public async Task OnLoadedAsync() =>
await (StartAsync?.InvokeAsync(this, EventArgs.Empty)).ConfigureAwait(false);
/// The server only processes any notifications *after* it has been initialized properly.
/// Hence we need to wait until after initialization to send all solution events that have already been raised,
/// and start listening to new ones.
public Task OnServerInitializedAsync() =>
Task.Run(() => Telemetry.SendEvent(Telemetry.ExtensionEvent.LspReady));
/// We create a new pane to show the encountered exception.
public async Task OnServerInitializeFailedAsync(Exception ex)
{
await VisualStudio.Shell.ThreadHelper.JoinableTaskFactory.SwitchToMainThreadAsync();
var outWindow = GetGlobalService(typeof(SVsOutputWindow)) as IVsOutputWindow;
var windowTitle = this.Name;
outWindow.CreatePane(ref OutputPaneGuid, windowTitle, 1, 1);
outWindow.GetPane(ref OutputPaneGuid, out var customPane);
var messages = new[]
{
$"Server initialization failed.",
$"Path to the language server executable: \"{this.LanguageServerPath}\"",
$"Path to the log file: \"{this.LogFile}\"",
ex.ToString()
};
customPane.OutputStringThreadSafe(String.Join(Environment.NewLine, messages));
customPane.Activate(); // brings the pane into view
}
public Task<InitializationFailureContext> OnServerInitializeFailedAsync(ILanguageClientInitializationInfo initializationState)
{
string message = "Q# Language Client failed to activate.";
string exception = initializationState.InitializationException?.ToString() ?? string.Empty;
message = $"{message}\n {exception}";
var failureContext = new InitializationFailureContext()
{
FailureMessage = message,
};
return Task.FromResult<InitializationFailureContext>(failureContext);
}
/// Invoking the StartAsync event signals that the language server should be started, and triggers a call to this routine.
/// This routine contains the logic to start the server and estabilishes a connection that is returned (exceptions here are shown in the info bar of VS).
public async Task<Connection> ActivateAsync(CancellationToken token)
{
await Task.Yield();
Telemetry.SendEvent(Telemetry.ExtensionEvent.Activate);
try
{
#if MANUAL
string ServerReaderPipe = $"QsLanguageServerReaderPipe";
string ServerWriterPipe = $"QsLanguageServerWriterPipe";
#else
string ServerReaderPipe = $"QsLanguageServerReaderPipe{this.LaunchTime}";
string ServerWriterPipe = $"QsLanguageServerWriterPipe{this.LaunchTime}";
ProcessStartInfo info = new ProcessStartInfo
{
FileName = LanguageServerPath,
UseShellExecute = false,
CreateNoWindow = true,
Arguments = $"--writer={ServerWriterPipe} --reader={ServerReaderPipe} --log={this.LogFile}"
};
Process process = new Process { StartInfo = info };
if (!process.Start()) return null;
#endif
var bufferSize = 256;
var pipeSecurity = new PipeSecurity();
var id = new SecurityIdentifier(WellKnownSidType.AuthenticatedUserSid, null); // I don't think WorldSid ("Everyone") is necessary
pipeSecurity.AddAccessRule(new PipeAccessRule(id, PipeAccessRights.ReadWrite, System.Security.AccessControl.AccessControlType.Allow));
var readerPipe = new NamedPipeServerStream(ServerWriterPipe, PipeDirection.InOut, 4, PipeTransmissionMode.Message, PipeOptions.Asynchronous, bufferSize, bufferSize, pipeSecurity);
var writerPipe = new NamedPipeServerStream(ServerReaderPipe, PipeDirection.InOut, 4, PipeTransmissionMode.Message, PipeOptions.Asynchronous, bufferSize, bufferSize, pipeSecurity);
await readerPipe.WaitForConnectionAsync(token).ConfigureAwait(true);
await writerPipe.WaitForConnectionAsync(token).ConfigureAwait(true);
return new Connection(readerPipe, writerPipe);
}
catch (Exception e)
{
Telemetry.SendEvent(Telemetry.ExtensionEvent.Error, ("id", e.GetType().Name), ("reason", e.Message));
throw;
}
}
/// The list of server notifications not part of the default LanguageServer Protocol:
class CustomServerNotifications
{
[JsonRpcMethod("telemetry/event")]
public void OnTelemetry(JToken args)
{
try
{
var name = (string)args["event"];
var props = args["properties"]?.ToObject<Dictionary<string, object>>();
Telemetry.SendEvent(name, props);
}
catch (Exception ex)
{ Debug.Assert(false, $"error sending telemetry: \n{ex}"); }
}
}
/// Setup to get the VS version number including the correct minor version,
/// since the DTE.Version does not seem to accurately reflect that.
private static string GetVisualStudioVersion()
{
FileVersionInfo versionInfo;
try
{
var msenvPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "msenv.dll");
versionInfo = GetVersionInfo(msenvPath);
}
catch (FileNotFoundException)
{ return null; }
// Extract the version number from the string in the format "D16.2", "D16.3", etc.
var version = Regex.Match(versionInfo.FileVersion, @"D([\d\.]+)");
return version.Success ? version.Groups[1].Value : null;
}
}
}