|
|
@@ -2,9 +2,8 @@
|
|
|
// Exceptions. See /LICENSE for license information.
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
|
|
-#include "language_server/language_server.h"
|
|
|
+#include "toolchain/language_server/server.h"
|
|
|
|
|
|
-#include "clang-tools-extra/clangd/Protocol.h"
|
|
|
#include "toolchain/base/value_store.h"
|
|
|
#include "toolchain/diagnostics/null_diagnostics.h"
|
|
|
#include "toolchain/lex/lex.h"
|
|
|
@@ -13,22 +12,48 @@
|
|
|
#include "toolchain/parse/tree_and_subtrees.h"
|
|
|
#include "toolchain/source/source_buffer.h"
|
|
|
|
|
|
-namespace Carbon::LS {
|
|
|
+namespace Carbon::LanguageServer {
|
|
|
+
|
|
|
+Server::Server(std::FILE* input_stream, llvm::raw_ostream& output_stream)
|
|
|
+ : transport_(clang::clangd::newJSONTransport(input_stream, output_stream,
|
|
|
+ /*InMirror=*/nullptr,
|
|
|
+ /*Pretty=*/true)),
|
|
|
+ binder_(handlers_, *this) {
|
|
|
+ binder_.notification("textDocument/didOpen", this,
|
|
|
+ &Server::OnDidOpenTextDocument);
|
|
|
+ binder_.notification("textDocument/didChange", this,
|
|
|
+ &Server::OnDidChangeTextDocument);
|
|
|
+ binder_.method("initialize", this, &Server::OnInitialize);
|
|
|
+ binder_.method("textDocument/documentSymbol", this,
|
|
|
+ &Server::OnDocumentSymbol);
|
|
|
+}
|
|
|
+
|
|
|
+auto Server::Run() -> ErrorOr<Success> {
|
|
|
+ llvm::Error err = transport_->loop(*this);
|
|
|
+ if (err.success()) {
|
|
|
+ return Success();
|
|
|
+ } else {
|
|
|
+ std::string str;
|
|
|
+ llvm::raw_string_ostream out(str);
|
|
|
+ out << err;
|
|
|
+ return Error(str);
|
|
|
+ }
|
|
|
+}
|
|
|
|
|
|
-void LanguageServer::OnDidOpenTextDocument(
|
|
|
+void Server::OnDidOpenTextDocument(
|
|
|
clang::clangd::DidOpenTextDocumentParams const& params) {
|
|
|
files_.emplace(params.textDocument.uri.file(), params.textDocument.text);
|
|
|
}
|
|
|
|
|
|
-void LanguageServer::OnDidChangeTextDocument(
|
|
|
+void Server::OnDidChangeTextDocument(
|
|
|
clang::clangd::DidChangeTextDocumentParams const& params) {
|
|
|
- // full text is sent if full sync is specified in capabilities.
|
|
|
- assert(params.contentChanges.size() == 1);
|
|
|
+ // Full text is sent if full sync is specified in capabilities.
|
|
|
+ CARBON_CHECK(params.contentChanges.size() == 1);
|
|
|
std::string file = params.textDocument.uri.file().str();
|
|
|
files_[file] = params.contentChanges[0].text;
|
|
|
}
|
|
|
|
|
|
-void LanguageServer::OnInitialize(
|
|
|
+void Server::OnInitialize(
|
|
|
clang::clangd::NoParams const& /*client_capabilities*/,
|
|
|
clang::clangd::Callback<llvm::json::Object> cb) {
|
|
|
llvm::json::Object capabilities{{"documentSymbolProvider", true},
|
|
|
@@ -38,8 +63,7 @@ void LanguageServer::OnInitialize(
|
|
|
cb(reply);
|
|
|
}
|
|
|
|
|
|
-auto LanguageServer::onNotify(llvm::StringRef method, llvm::json::Value value)
|
|
|
- -> bool {
|
|
|
+auto Server::onNotify(llvm::StringRef method, llvm::json::Value value) -> bool {
|
|
|
if (method == "exit") {
|
|
|
return false;
|
|
|
}
|
|
|
@@ -53,8 +77,8 @@ auto LanguageServer::onNotify(llvm::StringRef method, llvm::json::Value value)
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
-auto LanguageServer::onCall(llvm::StringRef method, llvm::json::Value params,
|
|
|
- llvm::json::Value id) -> bool {
|
|
|
+auto Server::onCall(llvm::StringRef method, llvm::json::Value params,
|
|
|
+ llvm::json::Value id) -> bool {
|
|
|
if (auto handler = handlers_.MethodHandlers.find(method);
|
|
|
handler != handlers_.MethodHandlers.end()) {
|
|
|
// TODO: improve this if add threads
|
|
|
@@ -71,9 +95,8 @@ auto LanguageServer::onCall(llvm::StringRef method, llvm::json::Value params,
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
-auto LanguageServer::onReply(llvm::json::Value /*id*/,
|
|
|
- llvm::Expected<llvm::json::Value> /*result*/)
|
|
|
- -> bool {
|
|
|
+auto Server::onReply(llvm::json::Value /*id*/,
|
|
|
+ llvm::Expected<llvm::json::Value> /*result*/) -> bool {
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
@@ -94,7 +117,7 @@ static auto GetIdentifierName(const SharedValueStores& value_stores,
|
|
|
return std::nullopt;
|
|
|
}
|
|
|
|
|
|
-void LanguageServer::OnDocumentSymbol(
|
|
|
+void Server::OnDocumentSymbol(
|
|
|
clang::clangd::DocumentSymbolParams const& params,
|
|
|
clang::clangd::Callback<std::vector<clang::clangd::DocumentSymbol>> cb) {
|
|
|
SharedValueStores value_stores;
|
|
|
@@ -148,19 +171,4 @@ void LanguageServer::OnDocumentSymbol(
|
|
|
cb(result);
|
|
|
}
|
|
|
|
|
|
-void LanguageServer::Start() {
|
|
|
- auto transport =
|
|
|
- clang::clangd::newJSONTransport(stdin, llvm::outs(), nullptr, true);
|
|
|
- LanguageServer ls(std::move(transport));
|
|
|
- clang::clangd::LSPBinder binder(ls.handlers_, ls);
|
|
|
- binder.notification("textDocument/didOpen", &ls,
|
|
|
- &LanguageServer::OnDidOpenTextDocument);
|
|
|
- binder.notification("textDocument/didChange", &ls,
|
|
|
- &LanguageServer::OnDidChangeTextDocument);
|
|
|
- binder.method("initialize", &ls, &LanguageServer::OnInitialize);
|
|
|
- binder.method("textDocument/documentSymbol", &ls,
|
|
|
- &LanguageServer::OnDocumentSymbol);
|
|
|
- auto error = ls.transport_->loop(ls);
|
|
|
- llvm::errs() << "Error: " << error << "\n";
|
|
|
-}
|
|
|
-} // namespace Carbon::LS
|
|
|
+} // namespace Carbon::LanguageServer
|