nixd
Loading...
Searching...
No Matches
Connection.cpp
Go to the documentation of this file.
2#include "lspserver/Logger.h"
4
5#include <llvm/ADT/SmallString.h>
6#include <llvm/Support/CommandLine.h>
7
8#include <poll.h>
9#include <sys/poll.h>
10#include <sys/stat.h>
11
12#include <csignal>
13#include <cstdint>
14#include <cstdio>
15#include <memory>
16#include <optional>
17#include <system_error>
18
19namespace {
20
21llvm::cl::opt<int> ClientProcessID{
22 "clientProcessId",
23 llvm::cl::desc(
24 "Client process ID, if this PID died, the server should exit."),
25 llvm::cl::init(getppid())};
26
27} // namespace
28
29namespace lspserver {
30
31static llvm::json::Object encodeError(llvm::Error Error) {
32 std::string Message;
34 auto HandlerFn = [&](const LSPError &E) -> llvm::Error {
35 Message = E.Message;
36 Code = E.Code;
37 return llvm::Error::success();
38 };
39 if (llvm::Error Unhandled = llvm::handleErrors(std::move(Error), HandlerFn))
40 Message = llvm::toString(std::move(Unhandled));
41
42 return llvm::json::Object{
43 {"message", std::move(Message)},
44 {"code", int64_t(Code)},
45 };
46}
47
48/// Decode the given JSON object into an error.
49llvm::Error decodeError(const llvm::json::Object &O) {
50 llvm::StringRef Message =
51 O.getString("message").value_or("Unspecified error");
52 if (std::optional<int64_t> Code = O.getInteger("code"))
53 return llvm::make_error<LSPError>(Message.str(), ErrorCode(*Code));
54 return llvm::make_error<llvm::StringError>(llvm::inconvertibleErrorCode(),
55 Message.str());
56}
57
58void OutboundPort::notify(llvm::StringRef Method, llvm::json::Value Params) {
59 sendMessage(llvm::json::Object{
60 {"jsonrpc", "2.0"},
61 {"method", Method},
62 {"params", std::move(Params)},
63 });
64}
65void OutboundPort::call(llvm::StringRef Method, llvm::json::Value Params,
66 llvm::json::Value ID) {
67 sendMessage(llvm::json::Object{
68 {"jsonrpc", "2.0"},
69 {"id", std::move(ID)},
70 {"method", Method},
71 {"params", std::move(Params)},
72 });
73}
74void OutboundPort::reply(llvm::json::Value ID,
75 llvm::Expected<llvm::json::Value> Result) {
76 if (Result) {
77 sendMessage(llvm::json::Object{
78 {"jsonrpc", "2.0"},
79 {"id", std::move(ID)},
80 {"result", std::move(*Result)},
81 });
82 } else {
83 sendMessage(llvm::json::Object{
84 {"jsonrpc", "2.0"},
85 {"id", std::move(ID)},
86 {"error", encodeError(Result.takeError())},
87 });
88 }
89}
90
91void OutboundPort::sendMessage(llvm::json::Value Message) {
92 // Make sure our outputs are not interleaving between messages (json)
93 vlog(">>> {0}", Message);
94 std::lock_guard<std::mutex> Guard(Mutex);
95 OutputBuffer.clear();
96 llvm::raw_svector_ostream SVecOS(OutputBuffer);
97 SVecOS << (Pretty ? llvm::formatv("{0:2}", Message)
98 : llvm::formatv("{0}", Message));
99 Outs << "Content-Length: " << OutputBuffer.size() << "\r\n\r\n"
100 << OutputBuffer;
101 Outs.flush();
102}
103
104bool InboundPort::dispatch(llvm::json::Value Message, MessageHandler &Handler) {
105 // Message must be an object with "jsonrpc":"2.0".
106 auto *Object = Message.getAsObject();
107 if (!Object ||
108 Object->getString("jsonrpc") != std::optional<llvm::StringRef>("2.0")) {
109 elog("Not a JSON-RPC 2.0 message: {0:2}", Message);
110 return false;
111 }
112 // ID may be any JSON value. If absent, this is a notification.
113 std::optional<llvm::json::Value> ID;
114 if (auto *I = Object->get("id"))
115 ID = std::move(*I);
116 auto Method = Object->getString("method");
117 if (!Method) { // This is a response.
118 if (!ID) {
119 elog("No method and no response ID: {0:2}", Message);
120 return false;
121 }
122 if (auto *Err = Object->getObject("error"))
123 // TODO: Logging & reply errors.
124 return Handler.onReply(std::move(*ID), decodeError(*Err));
125 // Result should be given, use null if not.
126 llvm::json::Value Result = nullptr;
127 if (auto *R = Object->get("result"))
128 Result = std::move(*R);
129 return Handler.onReply(std::move(*ID), std::move(Result));
130 }
131 // Params should be given, use null if not.
132 llvm::json::Value Params = nullptr;
133 if (auto *P = Object->get("params"))
134 Params = std::move(*P);
135
136 if (ID)
137 return Handler.onCall(*Method, std::move(Params), std::move(*ID));
138 return Handler.onNotify(*Method, std::move(Params));
139}
140
141bool readLine(int fd, const std::atomic<bool> &Close,
142 llvm::SmallString<128> &Line) {
143 Line.clear();
144
145 std::vector<pollfd> FDs;
146 FDs.emplace_back(pollfd{
147 fd,
148 POLLIN | POLLPRI,
149 0,
150 });
151 for (;;) {
152 char Ch;
153 // FIXME: inefficient
154 int Poll = poll(FDs.data(), FDs.size(), 1000);
155 if (Poll < 0)
156 return false;
157 if (Close)
158 return false;
159
160 // POLLHUB: hang up from the writer side
161 // POLLNVAL: invalid request (fd not open)
162 if (FDs[0].revents & (POLLHUP | POLLNVAL)) {
163 return false;
164 }
165
166 if (FDs[0].revents & POLLIN) {
167 ssize_t BytesRead = read(fd, &Ch, 1);
168 if (BytesRead == -1) {
169 if (errno != EINTR)
170 return false;
171 } else if (BytesRead == 0)
172 return false;
173 else {
174 if (Ch == '\n')
175 return true;
176 Line += Ch;
177 }
178 }
179
180 if (kill(ClientProcessID, 0) < 0) {
181 // Parent died.
182 return false;
183 }
184 }
185}
186
187bool InboundPort::readStandardMessage(std::string &JSONString) {
188 unsigned long long ContentLength = 0;
189 llvm::SmallString<128> Line;
190 while (true) {
191 if (!readLine(In, Close, Line))
192 return false;
193
194 llvm::StringRef LineRef = Line;
195
196 // Content-Length is a mandatory header, and the only one we handle.
197 if (LineRef.consume_front("Content-Length: ")) {
198 llvm::getAsUnsignedInteger(LineRef.trim(), 0, ContentLength);
199 continue;
200 }
201 // An empty line indicates the end of headers.
202 // Go ahead and read the JSON.
203 if (LineRef.trim().empty())
204 break;
205 // It's another header, ignore it.
206 }
207
208 JSONString.resize(ContentLength);
209 for (size_t Pos = 0, Read; Pos < ContentLength; Pos += Read) {
210
211 Read = read(In, JSONString.data() + Pos, ContentLength - Pos);
212
213 if (Read == 0) {
214 elog("Input was aborted. Read only {0} bytes of expected {1}.", Pos,
215 ContentLength);
216 return false;
217 }
218 }
219 return true;
220}
221
222bool InboundPort::readDelimitedMessage(std::string &JSONString) {
223 JSONString.clear();
224 llvm::SmallString<128> Line;
225 bool IsInputBlock = false;
226 while (readLine(In, Close, Line)) {
227 auto LineRef = Line.str().trim();
228 if (IsInputBlock) {
229 // We are in input blocks, read lines and append JSONString.
230 if (LineRef.starts_with("#")) // comment
231 continue;
232
233 // End of the block
234 if (LineRef.starts_with("```")) {
235 IsInputBlock = false;
236 break;
237 }
238
239 JSONString += Line;
240 } else {
241 if (LineRef.starts_with("```json"))
242 IsInputBlock = true;
243 }
244 }
245 return true; // Including at EOF
246}
247
248bool InboundPort::readMessage(std::string &JSONString) {
249 switch (StreamStyle) {
250
252 return readStandardMessage(JSONString);
254 return readDelimitedMessage(JSONString);
255 break;
256 }
257 assert(false && "Invalid stream style");
258 __builtin_unreachable();
259}
260
262 std::string JSONString;
263 llvm::SmallString<128> Line;
264
265 for (;;) {
266 if (readMessage(JSONString)) {
267 vlog("<<< {0}", JSONString);
268 if (auto ExpectedParsedJSON = llvm::json::parse(JSONString)) {
269 if (!dispatch(*ExpectedParsedJSON, Handler))
270 return;
271 } else {
272 auto Err = ExpectedParsedJSON.takeError();
273 elog("The received json cannot be parsed, reason: {0}", Err);
274 return;
275 }
276 } else {
277 return;
278 }
279 }
280}
281
282} // namespace lspserver
bool dispatch(llvm::json::Value Message, MessageHandler &Handler)
JSONStreamStyle StreamStyle
Definition Connection.h:41
bool readStandardMessage(std::string &JSONString)
bool readMessage(std::string &JSONString)
bool readDelimitedMessage(std::string &JSONString)
void loop(MessageHandler &Handler)
virtual bool onReply(llvm::json::Value ID, llvm::Expected< llvm::json::Value > Result)=0
virtual bool onNotify(llvm::StringRef Method, llvm::json::Value)=0
virtual bool onCall(llvm::StringRef Method, llvm::json::Value Params, llvm::json::Value ID)=0
void sendMessage(llvm::json::Value Message)
void notify(llvm::StringRef Method, llvm::json::Value Params)
void reply(llvm::json::Value ID, llvm::Expected< llvm::json::Value > Result)
void call(llvm::StringRef Method, llvm::json::Value Params, llvm::json::Value ID)
Whether current platform treats paths case insensitively.
Definition Connection.h:11
@ Error
An error message.
llvm::Error decodeError(const llvm::json::Object &O)
Decode the given JSON object into an error.
void elog(const char *Fmt, Ts &&...Vals)
Definition Logger.h:52
void vlog(const char *Fmt, Ts &&...Vals)
Definition Logger.h:63
bool readLine(int fd, const std::atomic< bool > &Close, llvm::SmallString< 128 > &Line)