You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			361 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			361 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===--- ClangdLSPServer.cpp - LSP server ------------------------*- C++-*-===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===---------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "ClangdLSPServer.h"
 | ||
|  | #include "JSONRPCDispatcher.h"
 | ||
|  | #include "SourceCode.h"
 | ||
|  | #include "llvm/Support/FormatVariadic.h"
 | ||
|  | 
 | ||
|  | using namespace clang::clangd; | ||
|  | using namespace clang; | ||
|  | 
 | ||
|  | namespace { | ||
|  | 
 | ||
|  | TextEdit replacementToEdit(StringRef Code, const tooling::Replacement &R) { | ||
|  |   Range ReplacementRange = { | ||
|  |       offsetToPosition(Code, R.getOffset()), | ||
|  |       offsetToPosition(Code, R.getOffset() + R.getLength())}; | ||
|  |   return {ReplacementRange, R.getReplacementText()}; | ||
|  | } | ||
|  | 
 | ||
|  | std::vector<TextEdit> | ||
|  | replacementsToEdits(StringRef Code, | ||
|  |                     const std::vector<tooling::Replacement> &Replacements) { | ||
|  |   // Turn the replacements into the format specified by the Language Server
 | ||
|  |   // Protocol. Fuse them into one big JSON array.
 | ||
|  |   std::vector<TextEdit> Edits; | ||
|  |   for (const auto &R : Replacements) | ||
|  |     Edits.push_back(replacementToEdit(Code, R)); | ||
|  |   return Edits; | ||
|  | } | ||
|  | 
 | ||
|  | std::vector<TextEdit> replacementsToEdits(StringRef Code, | ||
|  |                                           const tooling::Replacements &Repls) { | ||
|  |   std::vector<TextEdit> Edits; | ||
|  |   for (const auto &R : Repls) | ||
|  |     Edits.push_back(replacementToEdit(Code, R)); | ||
|  |   return Edits; | ||
|  | } | ||
|  | 
 | ||
|  | } // namespace
 | ||
|  | 
 | ||
|  | void ClangdLSPServer::onInitialize(Ctx C, InitializeParams &Params) { | ||
|  |   reply(C, json::obj{ | ||
|  |       {{"capabilities", | ||
|  |         json::obj{ | ||
|  |             {"textDocumentSync", 1}, | ||
|  |             {"documentFormattingProvider", true}, | ||
|  |             {"documentRangeFormattingProvider", true}, | ||
|  |             {"documentOnTypeFormattingProvider", | ||
|  |              json::obj{ | ||
|  |                  {"firstTriggerCharacter", "}"}, | ||
|  |                  {"moreTriggerCharacter", {}}, | ||
|  |              }}, | ||
|  |             {"codeActionProvider", true}, | ||
|  |             {"completionProvider", | ||
|  |              json::obj{ | ||
|  |                  {"resolveProvider", false}, | ||
|  |                  {"triggerCharacters", {".", ">", ":"}}, | ||
|  |              }}, | ||
|  |             {"signatureHelpProvider", | ||
|  |              json::obj{ | ||
|  |                  {"triggerCharacters", {"(", ","}}, | ||
|  |              }}, | ||
|  |             {"definitionProvider", true}, | ||
|  |             {"documentHighlightProvider", true}, | ||
|  |             {"renameProvider", true}, | ||
|  |             {"executeCommandProvider", | ||
|  |              json::obj{ | ||
|  |                  {"commands", {ExecuteCommandParams::CLANGD_APPLY_FIX_COMMAND}}, | ||
|  |              }}, | ||
|  |         }}}}); | ||
|  |   if (Params.rootUri && !Params.rootUri->file.empty()) | ||
|  |     Server.setRootPath(Params.rootUri->file); | ||
|  |   else if (Params.rootPath && !Params.rootPath->empty()) | ||
|  |     Server.setRootPath(*Params.rootPath); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onShutdown(Ctx C, ShutdownParams &Params) { | ||
|  |   // Do essentially nothing, just say we're ready to exit.
 | ||
|  |   ShutdownRequestReceived = true; | ||
|  |   reply(C, nullptr); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onExit(Ctx C, ExitParams &Params) { IsDone = true; } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDocumentDidOpen(Ctx C, | ||
|  |                                         DidOpenTextDocumentParams &Params) { | ||
|  |   if (Params.metadata && !Params.metadata->extraFlags.empty()) | ||
|  |     CDB.setExtraFlagsForFile(Params.textDocument.uri.file, | ||
|  |                              std::move(Params.metadata->extraFlags)); | ||
|  |   Server.addDocument(std::move(C), Params.textDocument.uri.file, | ||
|  |                      Params.textDocument.text); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDocumentDidChange(Ctx C, | ||
|  |                                           DidChangeTextDocumentParams &Params) { | ||
|  |   if (Params.contentChanges.size() != 1) | ||
|  |     return replyError(C, ErrorCode::InvalidParams, | ||
|  |                       "can only apply one change at a time"); | ||
|  |   // We only support full syncing right now.
 | ||
|  |   Server.addDocument(std::move(C), Params.textDocument.uri.file, | ||
|  |                      Params.contentChanges[0].text); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onFileEvent(Ctx C, DidChangeWatchedFilesParams &Params) { | ||
|  |   Server.onFileEvent(Params); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onCommand(Ctx C, ExecuteCommandParams &Params) { | ||
|  |   if (Params.command == ExecuteCommandParams::CLANGD_APPLY_FIX_COMMAND && | ||
|  |       Params.workspaceEdit) { | ||
|  |     // The flow for "apply-fix" :
 | ||
|  |     // 1. We publish a diagnostic, including fixits
 | ||
|  |     // 2. The user clicks on the diagnostic, the editor asks us for code actions
 | ||
|  |     // 3. We send code actions, with the fixit embedded as context
 | ||
|  |     // 4. The user selects the fixit, the editor asks us to apply it
 | ||
|  |     // 5. We unwrap the changes and send them back to the editor
 | ||
|  |     // 6. The editor applies the changes (applyEdit), and sends us a reply (but
 | ||
|  |     // we ignore it)
 | ||
|  | 
 | ||
|  |     ApplyWorkspaceEditParams ApplyEdit; | ||
|  |     ApplyEdit.edit = *Params.workspaceEdit; | ||
|  |     reply(C, "Fix applied."); | ||
|  |     // We don't need the response so id == 1 is OK.
 | ||
|  |     // Ideally, we would wait for the response and if there is no error, we
 | ||
|  |     // would reply success/failure to the original RPC.
 | ||
|  |     call(C, "workspace/applyEdit", ApplyEdit); | ||
|  |   } else { | ||
|  |     // We should not get here because ExecuteCommandParams would not have
 | ||
|  |     // parsed in the first place and this handler should not be called. But if
 | ||
|  |     // more commands are added, this will be here has a safe guard.
 | ||
|  |     replyError( | ||
|  |         C, ErrorCode::InvalidParams, | ||
|  |         llvm::formatv("Unsupported command \"{0}\".", Params.command).str()); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onRename(Ctx C, RenameParams &Params) { | ||
|  |   auto File = Params.textDocument.uri.file; | ||
|  |   auto Replacements = Server.rename(C, File, Params.position, Params.newName); | ||
|  |   if (!Replacements) { | ||
|  |     replyError(C, ErrorCode::InternalError, | ||
|  |                llvm::toString(Replacements.takeError())); | ||
|  |     return; | ||
|  |   } | ||
|  |   std::string Code = Server.getDocument(File); | ||
|  |   std::vector<TextEdit> Edits = replacementsToEdits(Code, *Replacements); | ||
|  |   WorkspaceEdit WE; | ||
|  |   WE.changes = {{Params.textDocument.uri.uri, Edits}}; | ||
|  |   reply(C, WE); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDocumentDidClose(Ctx C, | ||
|  |                                          DidCloseTextDocumentParams &Params) { | ||
|  |   Server.removeDocument(std::move(C), Params.textDocument.uri.file); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDocumentOnTypeFormatting( | ||
|  |     Ctx C, DocumentOnTypeFormattingParams &Params) { | ||
|  |   auto File = Params.textDocument.uri.file; | ||
|  |   std::string Code = Server.getDocument(File); | ||
|  |   auto ReplacementsOrError = Server.formatOnType(Code, File, Params.position); | ||
|  |   if (ReplacementsOrError) | ||
|  |     reply(C, json::ary(replacementsToEdits(Code, ReplacementsOrError.get()))); | ||
|  |   else | ||
|  |     replyError(C, ErrorCode::UnknownErrorCode, | ||
|  |                llvm::toString(ReplacementsOrError.takeError())); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDocumentRangeFormatting( | ||
|  |     Ctx C, DocumentRangeFormattingParams &Params) { | ||
|  |   auto File = Params.textDocument.uri.file; | ||
|  |   std::string Code = Server.getDocument(File); | ||
|  |   auto ReplacementsOrError = Server.formatRange(Code, File, Params.range); | ||
|  |   if (ReplacementsOrError) | ||
|  |     reply(C, json::ary(replacementsToEdits(Code, ReplacementsOrError.get()))); | ||
|  |   else | ||
|  |     replyError(C, ErrorCode::UnknownErrorCode, | ||
|  |                llvm::toString(ReplacementsOrError.takeError())); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDocumentFormatting(Ctx C, | ||
|  |                                            DocumentFormattingParams &Params) { | ||
|  |   auto File = Params.textDocument.uri.file; | ||
|  |   std::string Code = Server.getDocument(File); | ||
|  |   auto ReplacementsOrError = Server.formatFile(Code, File); | ||
|  |   if (ReplacementsOrError) | ||
|  |     reply(C, json::ary(replacementsToEdits(Code, ReplacementsOrError.get()))); | ||
|  |   else | ||
|  |     replyError(C, ErrorCode::UnknownErrorCode, | ||
|  |                llvm::toString(ReplacementsOrError.takeError())); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onCodeAction(Ctx C, CodeActionParams &Params) { | ||
|  |   // We provide a code action for each diagnostic at the requested location
 | ||
|  |   // which has FixIts available.
 | ||
|  |   std::string Code = Server.getDocument(Params.textDocument.uri.file); | ||
|  |   json::ary Commands; | ||
|  |   for (Diagnostic &D : Params.context.diagnostics) { | ||
|  |     auto Edits = getFixIts(Params.textDocument.uri.file, D); | ||
|  |     if (!Edits.empty()) { | ||
|  |       WorkspaceEdit WE; | ||
|  |       WE.changes = {{Params.textDocument.uri.uri, std::move(Edits)}}; | ||
|  |       Commands.push_back(json::obj{ | ||
|  |           {"title", llvm::formatv("Apply FixIt {0}", D.message)}, | ||
|  |           {"command", ExecuteCommandParams::CLANGD_APPLY_FIX_COMMAND}, | ||
|  |           {"arguments", {WE}}, | ||
|  |       }); | ||
|  |     } | ||
|  |   } | ||
|  |   reply(C, std::move(Commands)); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onCompletion(Ctx C, TextDocumentPositionParams &Params) { | ||
|  |   auto Reply = Server | ||
|  |                    .codeComplete(std::move(C), Params.textDocument.uri.file, | ||
|  |                                  Position{Params.position.line, | ||
|  |                                           Params.position.character}, | ||
|  |                                  CCOpts) | ||
|  |                    .get(); // FIXME(ibiryukov): This could be made async if we
 | ||
|  |                            // had an API that would allow to attach callbacks to
 | ||
|  |                            // futures returned by ClangdServer.
 | ||
|  | 
 | ||
|  |   // We have std::move'd from C, now restore it from response of codeComplete.
 | ||
|  |   C = std::move(Reply.first); | ||
|  |   auto List = std::move(Reply.second.Value); | ||
|  |   reply(C, List); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onSignatureHelp(Ctx C, | ||
|  |                                       TextDocumentPositionParams &Params) { | ||
|  |   auto SignatureHelp = Server.signatureHelp( | ||
|  |       C, Params.textDocument.uri.file, | ||
|  |       Position{Params.position.line, Params.position.character}); | ||
|  |   if (!SignatureHelp) | ||
|  |     return replyError(C, ErrorCode::InvalidParams, | ||
|  |                       llvm::toString(SignatureHelp.takeError())); | ||
|  |   reply(C, SignatureHelp->Value); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onGoToDefinition(Ctx C, | ||
|  |                                        TextDocumentPositionParams &Params) { | ||
|  |   auto Items = Server.findDefinitions( | ||
|  |       C, Params.textDocument.uri.file, | ||
|  |       Position{Params.position.line, Params.position.character}); | ||
|  |   if (!Items) | ||
|  |     return replyError(C, ErrorCode::InvalidParams, | ||
|  |                       llvm::toString(Items.takeError())); | ||
|  |   reply(C, json::ary(Items->Value)); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onSwitchSourceHeader(Ctx C, | ||
|  |                                            TextDocumentIdentifier &Params) { | ||
|  |   llvm::Optional<Path> Result = Server.switchSourceHeader(Params.uri.file); | ||
|  |   std::string ResultUri; | ||
|  |   reply(C, Result ? URI::fromFile(*Result).uri : ""); | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDocumentHighlight(Ctx C, | ||
|  |                                           TextDocumentPositionParams &Params) { | ||
|  | 
 | ||
|  |   auto Highlights = Server.findDocumentHighlights( | ||
|  |       C, Params.textDocument.uri.file, | ||
|  |       Position{Params.position.line, Params.position.character}); | ||
|  | 
 | ||
|  |   if (!Highlights) { | ||
|  |     replyError(C, ErrorCode::InternalError, | ||
|  |                llvm::toString(Highlights.takeError())); | ||
|  |     return; | ||
|  |   } | ||
|  | 
 | ||
|  |   reply(C, json::ary(Highlights->Value)); | ||
|  | } | ||
|  | 
 | ||
|  | ClangdLSPServer::ClangdLSPServer(JSONOutput &Out, unsigned AsyncThreadsCount, | ||
|  |                                  bool StorePreamblesInMemory, | ||
|  |                                  const clangd::CodeCompleteOptions &CCOpts, | ||
|  |                                  llvm::Optional<StringRef> ResourceDir, | ||
|  |                                  llvm::Optional<Path> CompileCommandsDir, | ||
|  |                                  bool BuildDynamicSymbolIndex) | ||
|  |     : Out(Out), CDB(std::move(CompileCommandsDir)), CCOpts(CCOpts), | ||
|  |       Server(CDB, /*DiagConsumer=*/*this, FSProvider, AsyncThreadsCount, | ||
|  |              StorePreamblesInMemory, BuildDynamicSymbolIndex, ResourceDir) {} | ||
|  | 
 | ||
|  | bool ClangdLSPServer::run(std::istream &In) { | ||
|  |   assert(!IsDone && "Run was called before"); | ||
|  | 
 | ||
|  |   // Set up JSONRPCDispatcher.
 | ||
|  |   JSONRPCDispatcher Dispatcher([](Context Ctx, const json::Expr &Params) { | ||
|  |     replyError(Ctx, ErrorCode::MethodNotFound, "method not found"); | ||
|  |   }); | ||
|  |   registerCallbackHandlers(Dispatcher, Out, /*Callbacks=*/*this); | ||
|  | 
 | ||
|  |   // Run the Language Server loop.
 | ||
|  |   runLanguageServerLoop(In, Out, Dispatcher, IsDone); | ||
|  | 
 | ||
|  |   // Make sure IsDone is set to true after this method exits to ensure assertion
 | ||
|  |   // at the start of the method fires if it's ever executed again.
 | ||
|  |   IsDone = true; | ||
|  | 
 | ||
|  |   return ShutdownRequestReceived; | ||
|  | } | ||
|  | 
 | ||
|  | std::vector<TextEdit> ClangdLSPServer::getFixIts(StringRef File, | ||
|  |                                                  const clangd::Diagnostic &D) { | ||
|  |   std::lock_guard<std::mutex> Lock(FixItsMutex); | ||
|  |   auto DiagToFixItsIter = FixItsMap.find(File); | ||
|  |   if (DiagToFixItsIter == FixItsMap.end()) | ||
|  |     return {}; | ||
|  | 
 | ||
|  |   const auto &DiagToFixItsMap = DiagToFixItsIter->second; | ||
|  |   auto FixItsIter = DiagToFixItsMap.find(D); | ||
|  |   if (FixItsIter == DiagToFixItsMap.end()) | ||
|  |     return {}; | ||
|  | 
 | ||
|  |   return FixItsIter->second; | ||
|  | } | ||
|  | 
 | ||
|  | void ClangdLSPServer::onDiagnosticsReady( | ||
|  |     PathRef File, Tagged<std::vector<DiagWithFixIts>> Diagnostics) { | ||
|  |   json::ary DiagnosticsJSON; | ||
|  | 
 | ||
|  |   DiagnosticToReplacementMap LocalFixIts; // Temporary storage
 | ||
|  |   for (auto &DiagWithFixes : Diagnostics.Value) { | ||
|  |     auto Diag = DiagWithFixes.Diag; | ||
|  |     DiagnosticsJSON.push_back(json::obj{ | ||
|  |         {"range", Diag.range}, | ||
|  |         {"severity", Diag.severity}, | ||
|  |         {"message", Diag.message}, | ||
|  |     }); | ||
|  |     // We convert to Replacements to become independent of the SourceManager.
 | ||
|  |     auto &FixItsForDiagnostic = LocalFixIts[Diag]; | ||
|  |     std::copy(DiagWithFixes.FixIts.begin(), DiagWithFixes.FixIts.end(), | ||
|  |               std::back_inserter(FixItsForDiagnostic)); | ||
|  |   } | ||
|  | 
 | ||
|  |   // Cache FixIts
 | ||
|  |   { | ||
|  |     // FIXME(ibiryukov): should be deleted when documents are removed
 | ||
|  |     std::lock_guard<std::mutex> Lock(FixItsMutex); | ||
|  |     FixItsMap[File] = LocalFixIts; | ||
|  |   } | ||
|  | 
 | ||
|  |   // Publish diagnostics.
 | ||
|  |   Out.writeMessage(json::obj{ | ||
|  |       {"jsonrpc", "2.0"}, | ||
|  |       {"method", "textDocument/publishDiagnostics"}, | ||
|  |       {"params", | ||
|  |        json::obj{ | ||
|  |            {"uri", URI::fromFile(File)}, | ||
|  |            {"diagnostics", std::move(DiagnosticsJSON)}, | ||
|  |        }}, | ||
|  |   }); | ||
|  | } |