ccls/src/lsp.hh

347 lines
9.1 KiB
C++
Raw Normal View History

2018-08-21 05:27:52 +00:00
/* Copyright 2017-2018 ccls Authors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
==============================================================================*/
2017-03-05 02:16:23 +00:00
#pragma once
2018-10-29 04:21:21 +00:00
#include "config.hh"
#include "serializer.hh"
2018-10-29 04:21:21 +00:00
#include "utils.hh"
2017-03-25 20:32:44 +00:00
#include <rapidjson/fwd.h>
#include <iosfwd>
2017-03-05 02:16:23 +00:00
#include <unordered_map>
namespace ccls {
struct RequestId {
// The client can send the request id as an int or a string. We should output
// the same format we received.
enum Type { kNone, kInt, kString };
Type type = kNone;
int value = -1;
bool Valid() const { return type != kNone; }
};
void Reflect(Reader &visitor, RequestId &value);
void Reflect(Writer &visitor, RequestId &value);
struct InMessage {
RequestId id;
std::string method;
std::unique_ptr<char[]> message;
std::unique_ptr<rapidjson::Document> document;
2017-03-10 07:06:01 +00:00
};
2017-03-05 02:16:23 +00:00
enum class ErrorCode {
// Defined by JSON RPC
ParseError = -32700,
InvalidRequest = -32600,
MethodNotFound = -32601,
InvalidParams = -32602,
InternalError = -32603,
serverErrorStart = -32099,
serverErrorEnd = -32000,
ServerNotInitialized = -32002,
UnknownErrorCode = -32001,
// Defined by the protocol.
RequestCancelled = -32800,
2017-03-24 07:59:47 +00:00
};
MAKE_REFLECT_TYPE_PROXY(ErrorCode);
2017-03-24 07:59:47 +00:00
struct ResponseError {
// A number indicating the error type that occurred.
ErrorCode code;
// A string providing a short description of the error.
2017-03-24 07:59:47 +00:00
std::string message;
// A Primitive or Structured value that contains additional
// information about the error. Can be omitted.
// std::optional<D> data;
2017-03-24 07:59:47 +00:00
};
MAKE_REFLECT_STRUCT(ResponseError, code, message);
2017-03-24 07:59:47 +00:00
constexpr char ccls_xref[] = "ccls.xref";
constexpr char window_showMessage[] = "window/showMessage";
2017-03-10 07:06:01 +00:00
/////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
////////////////////////////// PRIMITIVE TYPES //////////////////////////////
/////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
2017-03-05 02:16:23 +00:00
struct DocumentUri {
static DocumentUri FromPath(const std::string &path);
2017-03-11 08:07:32 +00:00
bool operator==(const DocumentUri &other) const;
2017-03-25 23:58:11 +00:00
2018-08-09 17:08:14 +00:00
void SetPath(const std::string &path);
std::string GetPath() const;
2017-03-05 02:16:23 +00:00
std::string raw_uri;
2017-03-10 07:06:01 +00:00
};
2017-03-05 02:16:23 +00:00
2017-09-22 01:14:57 +00:00
template <typename TVisitor>
void Reflect(TVisitor &visitor, DocumentUri &value) {
2017-03-10 07:06:01 +00:00
Reflect(visitor, value.raw_uri);
}
2017-03-05 02:16:23 +00:00
2017-03-10 07:06:01 +00:00
struct lsPosition {
int line = 0;
int character = 0;
2018-08-09 17:08:14 +00:00
bool operator==(const lsPosition &o) const {
2018-04-08 00:10:54 +00:00
return line == o.line && character == o.character;
}
2018-08-09 17:08:14 +00:00
bool operator<(const lsPosition &o) const {
2018-04-08 00:10:54 +00:00
return line != o.line ? line < o.line : character < o.character;
}
std::string ToString() const;
2017-03-10 07:06:01 +00:00
};
2017-03-25 23:58:11 +00:00
MAKE_REFLECT_STRUCT(lsPosition, line, character);
2017-03-05 02:16:23 +00:00
2017-03-10 07:06:01 +00:00
struct lsRange {
lsPosition start;
lsPosition end;
2018-08-09 17:08:14 +00:00
bool operator==(const lsRange &o) const {
2018-04-08 00:10:54 +00:00
return start == o.start && end == o.end;
}
2018-08-09 17:08:14 +00:00
bool operator<(const lsRange &o) const {
2018-04-08 00:10:54 +00:00
return !(start == o.start) ? start < o.start : end < o.end;
}
2017-03-10 07:06:01 +00:00
};
2017-03-25 23:58:11 +00:00
MAKE_REFLECT_STRUCT(lsRange, start, end);
2017-03-05 02:16:23 +00:00
struct Location {
DocumentUri uri;
lsRange range;
bool operator==(const Location &o) const {
2018-04-08 00:10:54 +00:00
return uri == o.uri && range == o.range;
}
bool operator<(const Location &o) const {
2018-04-08 00:10:54 +00:00
return !(uri.raw_uri == o.uri.raw_uri) ? uri.raw_uri < o.uri.raw_uri
: range < o.range;
}
2017-03-10 07:06:01 +00:00
};
MAKE_REFLECT_STRUCT(Location, uri, range);
2017-03-05 02:16:23 +00:00
enum class SymbolKind : uint8_t {
2018-03-07 21:20:31 +00:00
Unknown = 0,
File = 1,
Module = 2,
Namespace = 3,
Package = 4,
Class = 5,
Method = 6,
Property = 7,
Field = 8,
Constructor = 9,
Enum = 10,
Interface = 11,
Function = 12,
Variable = 13,
Constant = 14,
String = 15,
Number = 16,
Boolean = 17,
Array = 18,
Object = 19,
Key = 20,
Null = 21,
EnumMember = 22,
Struct = 23,
Event = 24,
Operator = 25,
// For C++, this is interpreted as "template parameter" (including
// non-type template parameters).
TypeParameter = 26,
2018-03-31 03:16:33 +00:00
// ccls extensions
2018-03-07 21:20:31 +00:00
// See also https://github.com/Microsoft/language-server-protocol/issues/344
// for new SymbolKind clang/Index/IndexSymbol.h clang::index::SymbolKind
TypeAlias = 252,
Parameter = 253,
StaticMethod = 254,
Macro = 255,
};
MAKE_REFLECT_TYPE_PROXY(SymbolKind);
struct SymbolInformation {
2018-10-15 08:26:13 +00:00
std::string_view name;
SymbolKind kind;
Location location;
2018-10-15 08:26:13 +00:00
std::optional<std::string_view> containerName;
};
struct TextDocumentIdentifier {
DocumentUri uri;
2017-03-10 07:06:01 +00:00
};
MAKE_REFLECT_STRUCT(TextDocumentIdentifier, uri);
2017-03-05 02:16:23 +00:00
struct VersionedTextDocumentIdentifier {
DocumentUri uri;
// The version number of this document. number | null
2018-04-16 19:36:02 +00:00
std::optional<int> version;
};
MAKE_REFLECT_STRUCT(VersionedTextDocumentIdentifier, uri, version);
struct TextEdit {
2017-05-21 07:37:53 +00:00
// The range of the text document to be manipulated. To insert
// text into a document create a range where start === end.
lsRange range;
// The string to be inserted. For delete operations use an
// empty string.
std::string newText;
bool operator==(const TextEdit &that);
2017-05-21 07:37:53 +00:00
};
MAKE_REFLECT_STRUCT(TextEdit, range, newText);
2017-05-21 07:37:53 +00:00
struct TextDocumentItem {
// The text document's URI.
DocumentUri uri;
// The text document's language identifier.
std::string languageId;
// The version number of this document (it will strictly increase after each
// change, including undo/redo).
int version;
// The content of the opened text document.
std::string text;
};
MAKE_REFLECT_STRUCT(TextDocumentItem, uri, languageId, version, text);
struct TextDocumentEdit {
// The text document to change.
VersionedTextDocumentIdentifier textDocument;
// The edits to be applied.
std::vector<TextEdit> edits;
};
MAKE_REFLECT_STRUCT(TextDocumentEdit, textDocument, edits);
struct WorkspaceEdit {
std::vector<TextDocumentEdit> documentChanges;
};
MAKE_REFLECT_STRUCT(WorkspaceEdit, documentChanges);
struct TextDocumentContentChangeEvent {
2017-12-06 04:39:44 +00:00
// The range of the document that changed.
2018-03-31 03:16:33 +00:00
std::optional<lsRange> range;
2017-12-06 04:39:44 +00:00
// The length of the range that got replaced.
2018-03-31 03:16:33 +00:00
std::optional<int> rangeLength;
2017-12-06 04:39:44 +00:00
// The new text of the range/document.
std::string text;
2017-03-15 07:14:44 +00:00
};
struct TextDocumentDidChangeParam {
VersionedTextDocumentIdentifier textDocument;
std::vector<TextDocumentContentChangeEvent> contentChanges;
2017-03-10 07:06:01 +00:00
};
2017-03-06 08:48:51 +00:00
struct WorkspaceFolder {
DocumentUri uri;
2018-10-08 05:02:28 +00:00
std::string name;
};
MAKE_REFLECT_STRUCT(WorkspaceFolder, uri, name);
2018-10-08 05:02:28 +00:00
2017-03-25 21:45:49 +00:00
// Show a message to the user.
enum class MessageType : int { Error = 1, Warning = 2, Info = 3, Log = 4 };
MAKE_REFLECT_TYPE_PROXY(MessageType)
enum class DiagnosticSeverity {
// Reports an error.
Error = 1,
// Reports a warning.
Warning = 2,
// Reports an information.
Information = 3,
// Reports a hint.
Hint = 4
2017-03-10 07:06:01 +00:00
};
MAKE_REFLECT_TYPE_PROXY(DiagnosticSeverity);
struct Diagnostic {
// The range at which the message applies.
lsRange range;
// The diagnostic's severity. Can be omitted. If omitted it is up to the
// client to interpret diagnostics as error, warning, info or hint.
std::optional<DiagnosticSeverity> severity;
// The diagnostic's code. Can be omitted.
int code = 0;
// A human-readable string describing the source of this
// diagnostic, e.g. 'typescript' or 'super lint'.
std::string source = "ccls";
// The diagnostic's message.
std::string message;
2017-03-25 23:58:11 +00:00
// Non-serialized set of fixits.
std::vector<TextEdit> fixits_;
2017-03-10 07:06:01 +00:00
};
MAKE_REFLECT_STRUCT(Diagnostic, range, severity, source, message);
struct ShowMessageParam {
MessageType type = MessageType::Error;
std::string message;
};
MAKE_REFLECT_STRUCT(ShowMessageParam, type, message);
// Used to identify the language at a file level. The ordering is important, as
// a file previously identified as `C`, will be changed to `Cpp` if it
// encounters a c++ declaration.
enum class LanguageId { Unknown = -1, C = 0, Cpp = 1, ObjC = 2, ObjCpp = 3 };
MAKE_REFLECT_TYPE_PROXY(LanguageId);
2018-10-15 08:26:13 +00:00
// The order matters. In FindSymbolsAtLocation, we want Var/Func ordered in
// front of others.
enum class Kind : uint8_t { Invalid, File, Type, Func, Var };
MAKE_REFLECT_TYPE_PROXY(Kind);
2018-10-15 08:26:13 +00:00
enum class Role : uint16_t {
None = 0,
Declaration = 1 << 0,
Definition = 1 << 1,
Reference = 1 << 2,
Read = 1 << 3,
Write = 1 << 4,
Call = 1 << 5,
Dynamic = 1 << 6,
Address = 1 << 7,
Implicit = 1 << 8,
All = (1 << 9) - 1,
};
MAKE_REFLECT_TYPE_PROXY(Role);
inline uint16_t operator&(Role lhs, Role rhs) {
return uint16_t(lhs) & uint16_t(rhs);
}
inline Role operator|(Role lhs, Role rhs) {
return Role(uint16_t(lhs) | uint16_t(rhs));
}
} // namespace ccls