forked from OSchip/llvm-project
385 lines
14 KiB
C++
385 lines
14 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 "URI.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 Code = Server.getDocument(File);
|
|
if (!Code)
|
|
return replyError(C, ErrorCode::InvalidParams,
|
|
"onRename called for non-added file");
|
|
|
|
auto Replacements = Server.rename(C, File, Params.position, Params.newName);
|
|
if (!Replacements) {
|
|
replyError(C, ErrorCode::InternalError,
|
|
llvm::toString(Replacements.takeError()));
|
|
return;
|
|
}
|
|
|
|
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;
|
|
auto Code = Server.getDocument(File);
|
|
if (!Code)
|
|
return replyError(C, ErrorCode::InvalidParams,
|
|
"onDocumentOnTypeFormatting called for non-added 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;
|
|
auto Code = Server.getDocument(File);
|
|
if (!Code)
|
|
return replyError(C, ErrorCode::InvalidParams,
|
|
"onDocumentRangeFormatting called for non-added 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;
|
|
auto Code = Server.getDocument(File);
|
|
if (!Code)
|
|
return replyError(C, ErrorCode::InvalidParams,
|
|
"onDocumentFormatting called for non-added 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.
|
|
auto Code = Server.getDocument(Params.textDocument.uri.file);
|
|
if (!Code)
|
|
return replyError(C, ErrorCode::InvalidParams,
|
|
"onCodeAction called for non-added 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);
|
|
reply(C, Result ? URI::createFile(*Result).toString() : "");
|
|
}
|
|
|
|
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,
|
|
SymbolIndex *StaticIdx)
|
|
: Out(Out), CDB(std::move(CompileCommandsDir)), CCOpts(CCOpts),
|
|
Server(CDB, /*DiagConsumer=*/*this, FSProvider, AsyncThreadsCount,
|
|
StorePreamblesInMemory, BuildDynamicSymbolIndex, StaticIdx,
|
|
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(
|
|
const Context &Ctx, 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", URIForFile{File}},
|
|
{"diagnostics", std::move(DiagnosticsJSON)},
|
|
}},
|
|
});
|
|
}
|