1 /*
2  *Copyright (C) 2018 Laurent Tréguier
3  *
4  *This file is part of DLS.
5  *
6  *DLS is free software: you can redistribute it and/or modify
7  *it under the terms of the GNU General Public License as published by
8  *the Free Software Foundation, either version 3 of the License, or
9  *(at your option) any later version.
10  *
11  *DLS is distributed in the hope that it will be useful,
12  *but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *GNU General Public License for more details.
15  *
16  *You should have received a copy of the GNU General Public License
17  *along with DLS.  If not, see <http://www.gnu.org/licenses/>.
18  *
19  */
20 
21 module dls.protocol.messages.general;
22 
23 import dls.protocol.interfaces.general;
24 import std.json : JSONValue;
25 
26 @("")
27 InitializeResult initialize(InitializeParams params)
28 {
29     import dls.protocol.interfaces : CodeActionKind;
30     import dls.protocol.logger : logger;
31     import dls.protocol.state : initOptions, initState;
32     import dls.server : Server;
33     import dls.tools.analysis_tool : AnalysisTool;
34     import dls.tools.command_tool : CommandTool;
35     import dls.tools.format_tool : DfmtFormatTool, FormatTool;
36     import dls.tools.symbol_tool : SymbolTool;
37     import dls.tools.tool : Tool;
38     import dls.util.json : convertToJSON;
39     import dls.util.uri : Uri, filenameCmp, sameFile;
40     import std.algorithm : map, sort, uniq;
41     import std.array : array;
42     import std.typecons : Nullable, nullable;
43 
44     initState = params;
45     logger.info("Initializing server");
46     Tool.initialize(new Tool());
47     AnalysisTool.initialize(new AnalysisTool());
48     CommandTool.initialize(new CommandTool());
49     FormatTool.initialize(new DfmtFormatTool());
50     SymbolTool.initialize(new SymbolTool());
51 
52     debug
53     {
54     }
55     else
56     {
57         import dls.updater : cleanup;
58         import std.concurrency : spawn;
59 
60         spawn(&cleanup);
61     }
62 
63     Uri[] uris;
64 
65     if (!params.rootUri.isNull)
66     {
67         uris ~= new Uri(params.rootUri);
68     }
69     else if (!params.rootPath.isNull)
70     {
71         uris ~= Uri.fromPath(params.rootPath);
72     }
73 
74     if (!params.workspaceFolders.isNull)
75     {
76         uris ~= params.workspaceFolders.map!(wf => new Uri(wf.uri)).array;
77     }
78 
79     foreach (uri; uris.sort!((a, b) => filenameCmp(a, b) < 0)
80             .uniq!sameFile)
81     {
82         Tool.instance.updateConfig(uri, JSONValue());
83         SymbolTool.instance.importPath(uri);
84         AnalysisTool.instance.addAnalysisConfig(uri);
85     }
86 
87     auto result = new InitializeResult();
88 
89     with (result.capabilities)
90     {
91         import std.json : JSONValue;
92         import std.typecons : Nullable;
93 
94         textDocumentSync = new TextDocumentSyncOptions(
95             true.nullable,
96             TextDocumentSyncKind.incremental.nullable,
97             true.nullable,
98             false.nullable,
99             new SaveOptions(false.nullable).nullable
100         );
101 
102         if (!params.capabilities.textDocument.isNull)
103         {
104             const textDocCaps = params.capabilities.textDocument.get();
105 
106             if (!textDocCaps.synchronisation.isNull)
107             {
108                 if (!textDocCaps.synchronisation.willSave.isNull && textDocCaps.synchronisation.willSave.get())
109                 {
110                     textDocumentSync.save.nullify();
111                 }
112                 else if (!textDocCaps.synchronisation.didSave.isNull && textDocCaps.synchronisation.didSave.get())
113                 {
114                     textDocumentSync.willSave.nullify();
115                 }
116             }
117 
118             if (!textDocCaps.hover.isNull)
119                 hoverProvider = initOptions.capabilities.hover;
120 
121             if (!textDocCaps.completion.isNull && initOptions.capabilities.completion)
122                 completionProvider = new CompletionOptions(true.nullable, ["."].nullable);
123 
124             if (!textDocCaps.definition.isNull)
125                 definitionProvider = initOptions.capabilities.definition;
126 
127             if (!textDocCaps.typeDefinition.isNull)
128                 typeDefinitionProvider = JSONValue(initOptions.capabilities.definition);
129 
130             if (!textDocCaps.references.isNull)
131                 referencesProvider = initOptions.capabilities.references;
132 
133             if (!textDocCaps.documentHighlight.isNull)
134                 documentHighlightProvider = initOptions.capabilities.documentHighlight;
135 
136             if (!textDocCaps.documentSymbol.isNull)
137                 documentSymbolProvider = initOptions.capabilities.documentSymbol;
138 
139             if (!textDocCaps.codeAction.isNull)
140                 codeActionProvider = JSONValue(initOptions.capabilities.codeAction);
141 
142             if (!textDocCaps.formatting.isNull)
143                 documentFormattingProvider = initOptions.capabilities.documentFormatting;
144 
145             if (!textDocCaps.rangeFormatting.isNull)
146                 documentRangeFormattingProvider = initOptions.capabilities.documentRangeFormatting;
147 
148             if (!textDocCaps.onTypeFormatting.isNull
149                     && initOptions.capabilities.documentOnTypeFormatting)
150                 documentOnTypeFormattingProvider = new DocumentOnTypeFormattingOptions(";");
151 
152             if (!textDocCaps.rename.isNull)
153             {
154                 immutable prepareSupport = !textDocCaps.rename.prepareSupport.isNull
155                     && textDocCaps.rename.prepareSupport.get();
156 
157                 renameProvider = initOptions.capabilities.rename ? prepareSupport
158                     ? convertToJSON(new RenameOptions(true.nullable))
159                     : JSONValue(true) : JSONValue(false);
160             }
161         }
162 
163         if (!params.capabilities.workspace.isNull)
164         {
165             const workspaceCaps = params.capabilities.workspace.get();
166 
167             if (!workspaceCaps.symbol.isNull)
168                 workspaceSymbolProvider = initOptions.capabilities.workspaceSymbol;
169 
170             if (!workspaceCaps.executeCommand.isNull && initOptions.capabilities.codeAction)
171                 executeCommandProvider = new ExecuteCommandOptions(CommandTool.instance.commands);
172 
173             if (!workspaceCaps.workspaceFolders.isNull && workspaceCaps.workspaceFolders.get())
174                 workspace = new ServerCapabilities.Workspace(
175                         new ServerCapabilities.Workspace.WorkspaceFolders(true.nullable,
176                         JSONValue(true).nullable).nullable);
177         }
178     }
179 
180     Server.initialized = true;
181     return result;
182 }
183 
184 @("")
185 void initialized(JSONValue nothing)
186 {
187     import dls.protocol.interfaces : ConfigurationItem, ConfigurationParams,
188         DidChangeWatchedFilesRegistrationOptions, FileSystemWatcher,
189         Registration, RegistrationParams, WatchKind;
190     import dls.protocol.jsonrpc : send;
191     import dls.protocol.logger : logger;
192     import dls.protocol.messages.methods : Client, Workspace;
193     import dls.protocol.state : initOptions, initState;
194     import dls.server : Server;
195     import dls.tools.analysis_tool : AnalysisTool;
196     import dls.tools.tool : Tool;
197     import std.typecons : Nullable, nullable;
198 
199     debug
200     {
201     }
202     else
203     {
204         import dls.updater : update;
205         import std.concurrency : spawn;
206 
207         spawn(&update, initOptions.autoUpdate, initOptions.preReleaseBuilds);
208     }
209 
210     if (!initState.capabilities.workspace.isNull)
211     {
212         immutable didChangeWatchedFiles = !initState.capabilities.workspace.didChangeWatchedFiles.isNull
213             && initState.capabilities.workspace.didChangeWatchedFiles.dynamicRegistration;
214         ubyte watchAllEvents = WatchKind.create + WatchKind.change + WatchKind.delete_;
215 
216         if (didChangeWatchedFiles)
217         {
218             logger.info("Registering file watchers");
219             //dfmt off
220             auto watchers = [
221                 new FileSystemWatcher("**/dub.{json,sdl}", watchAllEvents.nullable),
222                 new FileSystemWatcher("**/dub.selections.json", watchAllEvents.nullable),
223                 new FileSystemWatcher("**/.gitmodules", watchAllEvents.nullable),
224                 new FileSystemWatcher("**/*.ini", watchAllEvents.nullable),
225                 new FileSystemWatcher("**/*.{d,di}", watchAllEvents.nullable)
226             ];
227             //dfmt on
228             auto registrationOptions = new DidChangeWatchedFilesRegistrationOptions(watchers);
229             auto registration = new Registration!DidChangeWatchedFilesRegistrationOptions("dls-file-watchers",
230                     "workspace/didChangeWatchedFiles", registrationOptions.nullable);
231             send(Client.registerCapability,
232                     new RegistrationParams!DidChangeWatchedFilesRegistrationOptions([registration]));
233         }
234 
235         immutable configuration = !initState.capabilities.workspace.configuration.isNull
236             && initState.capabilities.workspace.configuration;
237 
238         if (configuration)
239         {
240             auto items = [new ConfigurationItem(Nullable!string(null))];
241 
242             foreach (uri; Tool.instance.workspacesUris)
243             {
244                 items ~= new ConfigurationItem(uri.toString().nullable);
245             }
246 
247             send(Workspace.configuration, new ConfigurationParams(items));
248         }
249     }
250 
251     AnalysisTool.instance.scanAllWorkspaces();
252 }
253 
254 @("")
255 JSONValue shutdown(JSONValue nothing)
256 {
257     import dls.protocol.definitions : TextDocumentIdentifier;
258     import dls.protocol.logger : logger;
259     import dls.server : Server;
260     import dls.tools.analysis_tool : AnalysisTool;
261     import dls.tools.command_tool : CommandTool;
262     import dls.tools.format_tool : FormatTool;
263     import dls.tools.symbol_tool : SymbolTool;
264     import dls.tools.tool : Tool;
265     import dls.util.document : Document;
266 
267     logger.info("Shutting down server");
268     Server.initialized = false;
269     AnalysisTool.shutdown();
270     CommandTool.shutdown();
271     FormatTool.shutdown();
272     SymbolTool.shutdown();
273     Tool.shutdown();
274 
275     foreach (uri; Document.uris)
276     {
277         Document.close(new TextDocumentIdentifier(uri));
278     }
279 
280     return JSONValue(null);
281 }
282 
283 @("")
284 void exit(JSONValue nothing)
285 {
286     import dls.protocol.logger : logger;
287     import dls.server : Server;
288 
289     if (Server.initialized)
290     {
291         logger.warning("Shutdown not requested prior to exit");
292         shutdown(JSONValue());
293         Server.initialized = true;
294     }
295 
296     logger.info("Exiting server");
297     Server.exit = true;
298 }
299 
300 @("$")
301 void cancelRequest(CancelParams params)
302 {
303     import dls.server : Server;
304 
305     Server.cancel(params.id);
306 }