forked from OSchip/llvm-project
538 lines
15 KiB
C++
538 lines
15 KiB
C++
//===--- Protocol.cpp - Language Server Protocol Implementation -----------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file contains the serialization code for the LSP structs.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "Protocol.h"
|
|
#include "Logger.h"
|
|
#include "URI.h"
|
|
#include "clang/Basic/LLVM.h"
|
|
#include "llvm/ADT/SmallString.h"
|
|
#include "llvm/Support/Format.h"
|
|
#include "llvm/Support/FormatVariadic.h"
|
|
#include "llvm/Support/Path.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
namespace clang {
|
|
namespace clangd {
|
|
|
|
URIForFile::URIForFile(std::string AbsPath) {
|
|
assert(llvm::sys::path::is_absolute(AbsPath) && "the path is relative");
|
|
File = std::move(AbsPath);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &E, URIForFile &R) {
|
|
if (auto S = E.asString()) {
|
|
auto U = URI::parse(*S);
|
|
if (!U) {
|
|
log("Failed to parse URI " + *S + ": " + llvm::toString(U.takeError()));
|
|
return false;
|
|
}
|
|
if (U->scheme() != "file" && U->scheme() != "test") {
|
|
log("Clangd only supports 'file' URI scheme for workspace files: " + *S);
|
|
return false;
|
|
}
|
|
auto Path = URI::resolve(*U);
|
|
if (!Path) {
|
|
log(llvm::toString(Path.takeError()));
|
|
return false;
|
|
}
|
|
R = URIForFile(*Path);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
json::Expr toJSON(const URIForFile &U) { return U.uri(); }
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const URIForFile &U) {
|
|
return OS << U.uri();
|
|
}
|
|
|
|
json::Expr toJSON(const TextDocumentIdentifier &R) {
|
|
return json::obj{{"uri", R.uri}};
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, TextDocumentIdentifier &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("uri", R.uri);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, Position &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("line", R.line) && O.map("character", R.character);
|
|
}
|
|
|
|
json::Expr toJSON(const Position &P) {
|
|
return json::obj{
|
|
{"line", P.line},
|
|
{"character", P.character},
|
|
};
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const Position &P) {
|
|
return OS << P.line << ':' << P.character;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, Range &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("start", R.start) && O.map("end", R.end);
|
|
}
|
|
|
|
json::Expr toJSON(const Range &P) {
|
|
return json::obj{
|
|
{"start", P.start},
|
|
{"end", P.end},
|
|
};
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const Range &R) {
|
|
return OS << R.start << '-' << R.end;
|
|
}
|
|
|
|
json::Expr toJSON(const Location &P) {
|
|
return json::obj{
|
|
{"uri", P.uri},
|
|
{"range", P.range},
|
|
};
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const Location &L) {
|
|
return OS << L.range << '@' << L.uri;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, TextDocumentItem &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("uri", R.uri) && O.map("languageId", R.languageId) &&
|
|
O.map("version", R.version) && O.map("text", R.text);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, Metadata &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O)
|
|
return false;
|
|
O.map("extraFlags", R.extraFlags);
|
|
return true;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, TextEdit &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("range", R.range) && O.map("newText", R.newText);
|
|
}
|
|
|
|
json::Expr toJSON(const TextEdit &P) {
|
|
return json::obj{
|
|
{"range", P.range},
|
|
{"newText", P.newText},
|
|
};
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const TextEdit &TE) {
|
|
OS << TE.range << " => \"";
|
|
PrintEscapedString(TE.newText, OS);
|
|
return OS << '"';
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &E, TraceLevel &Out) {
|
|
if (auto S = E.asString()) {
|
|
if (*S == "off") {
|
|
Out = TraceLevel::Off;
|
|
return true;
|
|
} else if (*S == "messages") {
|
|
Out = TraceLevel::Messages;
|
|
return true;
|
|
} else if (*S == "verbose") {
|
|
Out = TraceLevel::Verbose;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, CompletionItemClientCapabilities &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O)
|
|
return false;
|
|
O.map("snippetSupport", R.snippetSupport);
|
|
O.map("commitCharacterSupport", R.commitCharacterSupport);
|
|
return true;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, CompletionClientCapabilities &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O)
|
|
return false;
|
|
O.map("dynamicRegistration", R.dynamicRegistration);
|
|
O.map("completionItem", R.completionItem);
|
|
O.map("contextSupport", R.contextSupport);
|
|
return true;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, TextDocumentClientCapabilities &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O)
|
|
return false;
|
|
O.map("completion", R.completion);
|
|
return true;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, ClientCapabilities &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O)
|
|
return false;
|
|
O.map("textDocument", R.textDocument);
|
|
return true;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, InitializeParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O)
|
|
return false;
|
|
// We deliberately don't fail if we can't parse individual fields.
|
|
// Failing to handle a slightly malformed initialize would be a disaster.
|
|
O.map("processId", R.processId);
|
|
O.map("rootUri", R.rootUri);
|
|
O.map("rootPath", R.rootPath);
|
|
O.map("capabilities", R.capabilities);
|
|
O.map("trace", R.trace);
|
|
// initializationOptions, capabilities unused
|
|
return true;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DidOpenTextDocumentParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("metadata", R.metadata);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DidCloseTextDocumentParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DidChangeTextDocumentParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("contentChanges", R.contentChanges) &&
|
|
O.map("wantDiagnostics", R.wantDiagnostics);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &E, FileChangeType &Out) {
|
|
if (auto T = E.asInteger()) {
|
|
if (*T < static_cast<int>(FileChangeType::Created) ||
|
|
*T > static_cast<int>(FileChangeType::Deleted))
|
|
return false;
|
|
Out = static_cast<FileChangeType>(*T);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, FileEvent &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("uri", R.uri) && O.map("type", R.type);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DidChangeWatchedFilesParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("changes", R.changes);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, TextDocumentContentChangeEvent &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("range", R.range) && O.map("rangeLength", R.rangeLength) &&
|
|
O.map("text", R.text);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, FormattingOptions &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("tabSize", R.tabSize) &&
|
|
O.map("insertSpaces", R.insertSpaces);
|
|
}
|
|
|
|
json::Expr toJSON(const FormattingOptions &P) {
|
|
return json::obj{
|
|
{"tabSize", P.tabSize},
|
|
{"insertSpaces", P.insertSpaces},
|
|
};
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DocumentRangeFormattingParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("range", R.range) && O.map("options", R.options);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DocumentOnTypeFormattingParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("position", R.position) && O.map("ch", R.ch) &&
|
|
O.map("options", R.options);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DocumentFormattingParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("options", R.options);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, Diagnostic &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O || !O.map("range", R.range) || !O.map("message", R.message))
|
|
return false;
|
|
O.map("severity", R.severity);
|
|
return true;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, CodeActionContext &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("diagnostics", R.diagnostics);
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, const Diagnostic &D) {
|
|
OS << D.range << " [";
|
|
switch (D.severity) {
|
|
case 1:
|
|
OS << "error";
|
|
break;
|
|
case 2:
|
|
OS << "warning";
|
|
break;
|
|
case 3:
|
|
OS << "note";
|
|
break;
|
|
case 4:
|
|
OS << "remark";
|
|
break;
|
|
default:
|
|
OS << "diagnostic";
|
|
break;
|
|
}
|
|
return OS << '(' << D.severity << "): " << D.message << "]";
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, CodeActionParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("range", R.range) && O.map("context", R.context);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, WorkspaceEdit &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("changes", R.changes);
|
|
}
|
|
|
|
const llvm::StringLiteral ExecuteCommandParams::CLANGD_APPLY_FIX_COMMAND =
|
|
"clangd.applyFix";
|
|
const llvm::StringLiteral ExecuteCommandParams::CLANGD_INSERT_HEADER_INCLUDE =
|
|
"clangd.insertInclude";
|
|
|
|
bool fromJSON(const json::Expr &Params, ExecuteCommandParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
if (!O || !O.map("command", R.command))
|
|
return false;
|
|
|
|
auto Args = Params.asObject()->getArray("arguments");
|
|
if (R.command == ExecuteCommandParams::CLANGD_APPLY_FIX_COMMAND) {
|
|
return Args && Args->size() == 1 &&
|
|
fromJSON(Args->front(), R.workspaceEdit);
|
|
} else if (R.command == ExecuteCommandParams::CLANGD_INSERT_HEADER_INCLUDE) {
|
|
return Args && Args->size() == 1 &&
|
|
fromJSON(Args->front(), R.includeInsertion);
|
|
}
|
|
return false; // Unrecognized command.
|
|
}
|
|
|
|
json::Expr toJSON(const Command &C) {
|
|
auto Cmd = json::obj{{"title", C.title}, {"command", C.command}};
|
|
if (C.workspaceEdit)
|
|
Cmd["arguments"] = {*C.workspaceEdit};
|
|
else if (C.includeInsertion)
|
|
Cmd["arguments"] = {*C.includeInsertion};
|
|
return std::move(Cmd);
|
|
}
|
|
|
|
json::Expr toJSON(const WorkspaceEdit &WE) {
|
|
if (!WE.changes)
|
|
return json::obj{};
|
|
json::obj FileChanges;
|
|
for (auto &Change : *WE.changes)
|
|
FileChanges[Change.first] = json::ary(Change.second);
|
|
return json::obj{{"changes", std::move(FileChanges)}};
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &II, IncludeInsertion &R) {
|
|
json::ObjectMapper O(II);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("declaringHeader", R.declaringHeader) &&
|
|
O.map("preferredHeader", R.preferredHeader);
|
|
}
|
|
json::Expr toJSON(const IncludeInsertion &II) {
|
|
return json::obj{{"textDocument", II.textDocument},
|
|
{"declaringHeader", II.declaringHeader},
|
|
{"preferredHeader", II.preferredHeader}};
|
|
}
|
|
|
|
json::Expr toJSON(const ApplyWorkspaceEditParams &Params) {
|
|
return json::obj{{"edit", Params.edit}};
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, TextDocumentPositionParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("position", R.position);
|
|
}
|
|
|
|
static StringRef toTextKind(MarkupKind Kind) {
|
|
switch (Kind) {
|
|
case MarkupKind::PlainText:
|
|
return "plaintext";
|
|
case MarkupKind::Markdown:
|
|
return "markdown";
|
|
}
|
|
llvm_unreachable("Invalid MarkupKind");
|
|
}
|
|
|
|
json::Expr toJSON(const MarkupContent &MC) {
|
|
if (MC.value.empty())
|
|
return nullptr;
|
|
|
|
return json::obj{
|
|
{"kind", toTextKind(MC.kind)},
|
|
{"value", MC.value},
|
|
};
|
|
}
|
|
|
|
json::Expr toJSON(const Hover &H) {
|
|
json::obj Result{{"contents", toJSON(H.contents)}};
|
|
|
|
if (H.range.hasValue())
|
|
Result["range"] = toJSON(*H.range);
|
|
|
|
return std::move(Result);
|
|
}
|
|
|
|
json::Expr toJSON(const CompletionItem &CI) {
|
|
assert(!CI.label.empty() && "completion item label is required");
|
|
json::obj Result{{"label", CI.label}};
|
|
if (CI.kind != CompletionItemKind::Missing)
|
|
Result["kind"] = static_cast<int>(CI.kind);
|
|
if (!CI.detail.empty())
|
|
Result["detail"] = CI.detail;
|
|
if (!CI.documentation.empty())
|
|
Result["documentation"] = CI.documentation;
|
|
if (!CI.sortText.empty())
|
|
Result["sortText"] = CI.sortText;
|
|
if (!CI.filterText.empty())
|
|
Result["filterText"] = CI.filterText;
|
|
if (!CI.insertText.empty())
|
|
Result["insertText"] = CI.insertText;
|
|
if (CI.insertTextFormat != InsertTextFormat::Missing)
|
|
Result["insertTextFormat"] = static_cast<int>(CI.insertTextFormat);
|
|
if (CI.textEdit)
|
|
Result["textEdit"] = *CI.textEdit;
|
|
if (!CI.additionalTextEdits.empty())
|
|
Result["additionalTextEdits"] = json::ary(CI.additionalTextEdits);
|
|
if (CI.command)
|
|
Result["command"] = *CI.command;
|
|
return std::move(Result);
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &O, const CompletionItem &I) {
|
|
O << I.label << " - " << toJSON(I);
|
|
return O;
|
|
}
|
|
|
|
bool operator<(const CompletionItem &L, const CompletionItem &R) {
|
|
return (L.sortText.empty() ? L.label : L.sortText) <
|
|
(R.sortText.empty() ? R.label : R.sortText);
|
|
}
|
|
|
|
json::Expr toJSON(const CompletionList &L) {
|
|
return json::obj{
|
|
{"isIncomplete", L.isIncomplete},
|
|
{"items", json::ary(L.items)},
|
|
};
|
|
}
|
|
|
|
json::Expr toJSON(const ParameterInformation &PI) {
|
|
assert(!PI.label.empty() && "parameter information label is required");
|
|
json::obj Result{{"label", PI.label}};
|
|
if (!PI.documentation.empty())
|
|
Result["documentation"] = PI.documentation;
|
|
return std::move(Result);
|
|
}
|
|
|
|
json::Expr toJSON(const SignatureInformation &SI) {
|
|
assert(!SI.label.empty() && "signature information label is required");
|
|
json::obj Result{
|
|
{"label", SI.label},
|
|
{"parameters", json::ary(SI.parameters)},
|
|
};
|
|
if (!SI.documentation.empty())
|
|
Result["documentation"] = SI.documentation;
|
|
return std::move(Result);
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &O,
|
|
const SignatureInformation &I) {
|
|
O << I.label << " - " << toJSON(I);
|
|
return O;
|
|
}
|
|
|
|
json::Expr toJSON(const SignatureHelp &SH) {
|
|
assert(SH.activeSignature >= 0 &&
|
|
"Unexpected negative value for number of active signatures.");
|
|
assert(SH.activeParameter >= 0 &&
|
|
"Unexpected negative value for active parameter index");
|
|
return json::obj{
|
|
{"activeSignature", SH.activeSignature},
|
|
{"activeParameter", SH.activeParameter},
|
|
{"signatures", json::ary(SH.signatures)},
|
|
};
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, RenameParams &R) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("textDocument", R.textDocument) &&
|
|
O.map("position", R.position) && O.map("newName", R.newName);
|
|
}
|
|
|
|
json::Expr toJSON(const DocumentHighlight &DH) {
|
|
return json::obj{
|
|
{"range", toJSON(DH.range)},
|
|
{"kind", static_cast<int>(DH.kind)},
|
|
};
|
|
}
|
|
|
|
llvm::raw_ostream &operator<<(llvm::raw_ostream &O,
|
|
const DocumentHighlight &V) {
|
|
O << V.range;
|
|
if (V.kind == DocumentHighlightKind::Read)
|
|
O << "(r)";
|
|
if (V.kind == DocumentHighlightKind::Write)
|
|
O << "(w)";
|
|
return O;
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, DidChangeConfigurationParams &CCP) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("settings", CCP.settings);
|
|
}
|
|
|
|
bool fromJSON(const json::Expr &Params, ClangdConfigurationParamsChange &CCPC) {
|
|
json::ObjectMapper O(Params);
|
|
return O && O.map("compilationDatabasePath", CCPC.compilationDatabasePath);
|
|
}
|
|
|
|
} // namespace clangd
|
|
} // namespace clang
|