annotate clang-tools-extra/clangd/JSONTransport.cpp @ 221:79ff65ed7e25

LLVM12 Original
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Tue, 15 Jun 2021 19:15:29 +0900
parents 0572611fdcc8
children 5f17cb93ff66
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 //===--- JSONTransport.cpp - sending and receiving LSP messages over JSON -===//
anatofuz
parents:
diff changeset
2 //
anatofuz
parents:
diff changeset
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
anatofuz
parents:
diff changeset
4 // See https://llvm.org/LICENSE.txt for license information.
anatofuz
parents:
diff changeset
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
anatofuz
parents:
diff changeset
6 //
anatofuz
parents:
diff changeset
7 //===----------------------------------------------------------------------===//
anatofuz
parents:
diff changeset
8 #include "Protocol.h" // For LSPError
anatofuz
parents:
diff changeset
9 #include "Transport.h"
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
10 #include "support/Cancellation.h"
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
11 #include "support/Logger.h"
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
12 #include "support/Shutdown.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
13 #include "llvm/ADT/SmallString.h"
150
anatofuz
parents:
diff changeset
14 #include "llvm/Support/Errno.h"
anatofuz
parents:
diff changeset
15 #include "llvm/Support/Error.h"
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
16 #include <system_error>
150
anatofuz
parents:
diff changeset
17
anatofuz
parents:
diff changeset
18 namespace clang {
anatofuz
parents:
diff changeset
19 namespace clangd {
anatofuz
parents:
diff changeset
20 namespace {
anatofuz
parents:
diff changeset
21
anatofuz
parents:
diff changeset
22 llvm::json::Object encodeError(llvm::Error E) {
anatofuz
parents:
diff changeset
23 std::string Message;
anatofuz
parents:
diff changeset
24 ErrorCode Code = ErrorCode::UnknownErrorCode;
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
25 // FIXME: encode cancellation errors using RequestCancelled or ContentModified
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
26 // as appropriate.
150
anatofuz
parents:
diff changeset
27 if (llvm::Error Unhandled = llvm::handleErrors(
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
28 std::move(E),
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
29 [&](const CancelledError &C) -> llvm::Error {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
30 switch (C.Reason) {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
31 case static_cast<int>(ErrorCode::ContentModified):
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
32 Code = ErrorCode::ContentModified;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
33 Message = "Request cancelled because the document was modified";
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
34 break;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
35 default:
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
36 Code = ErrorCode::RequestCancelled;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
37 Message = "Request cancelled";
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
38 break;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
39 }
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
40 return llvm::Error::success();
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
41 },
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
42 [&](const LSPError &L) -> llvm::Error {
150
anatofuz
parents:
diff changeset
43 Message = L.Message;
anatofuz
parents:
diff changeset
44 Code = L.Code;
anatofuz
parents:
diff changeset
45 return llvm::Error::success();
anatofuz
parents:
diff changeset
46 }))
anatofuz
parents:
diff changeset
47 Message = llvm::toString(std::move(Unhandled));
anatofuz
parents:
diff changeset
48
anatofuz
parents:
diff changeset
49 return llvm::json::Object{
anatofuz
parents:
diff changeset
50 {"message", std::move(Message)},
anatofuz
parents:
diff changeset
51 {"code", int64_t(Code)},
anatofuz
parents:
diff changeset
52 };
anatofuz
parents:
diff changeset
53 }
anatofuz
parents:
diff changeset
54
anatofuz
parents:
diff changeset
55 llvm::Error decodeError(const llvm::json::Object &O) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
56 llvm::StringRef Msg = O.getString("message").getValueOr("Unspecified error");
150
anatofuz
parents:
diff changeset
57 if (auto Code = O.getInteger("code"))
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
58 return llvm::make_error<LSPError>(Msg.str(), ErrorCode(*Code));
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
59 return error(Msg.str());
150
anatofuz
parents:
diff changeset
60 }
anatofuz
parents:
diff changeset
61
anatofuz
parents:
diff changeset
62 class JSONTransport : public Transport {
anatofuz
parents:
diff changeset
63 public:
anatofuz
parents:
diff changeset
64 JSONTransport(std::FILE *In, llvm::raw_ostream &Out,
anatofuz
parents:
diff changeset
65 llvm::raw_ostream *InMirror, bool Pretty, JSONStreamStyle Style)
anatofuz
parents:
diff changeset
66 : In(In), Out(Out), InMirror(InMirror ? *InMirror : llvm::nulls()),
anatofuz
parents:
diff changeset
67 Pretty(Pretty), Style(Style) {}
anatofuz
parents:
diff changeset
68
anatofuz
parents:
diff changeset
69 void notify(llvm::StringRef Method, llvm::json::Value Params) override {
anatofuz
parents:
diff changeset
70 sendMessage(llvm::json::Object{
anatofuz
parents:
diff changeset
71 {"jsonrpc", "2.0"},
anatofuz
parents:
diff changeset
72 {"method", Method},
anatofuz
parents:
diff changeset
73 {"params", std::move(Params)},
anatofuz
parents:
diff changeset
74 });
anatofuz
parents:
diff changeset
75 }
anatofuz
parents:
diff changeset
76 void call(llvm::StringRef Method, llvm::json::Value Params,
anatofuz
parents:
diff changeset
77 llvm::json::Value ID) override {
anatofuz
parents:
diff changeset
78 sendMessage(llvm::json::Object{
anatofuz
parents:
diff changeset
79 {"jsonrpc", "2.0"},
anatofuz
parents:
diff changeset
80 {"id", std::move(ID)},
anatofuz
parents:
diff changeset
81 {"method", Method},
anatofuz
parents:
diff changeset
82 {"params", std::move(Params)},
anatofuz
parents:
diff changeset
83 });
anatofuz
parents:
diff changeset
84 }
anatofuz
parents:
diff changeset
85 void reply(llvm::json::Value ID,
anatofuz
parents:
diff changeset
86 llvm::Expected<llvm::json::Value> Result) override {
anatofuz
parents:
diff changeset
87 if (Result) {
anatofuz
parents:
diff changeset
88 sendMessage(llvm::json::Object{
anatofuz
parents:
diff changeset
89 {"jsonrpc", "2.0"},
anatofuz
parents:
diff changeset
90 {"id", std::move(ID)},
anatofuz
parents:
diff changeset
91 {"result", std::move(*Result)},
anatofuz
parents:
diff changeset
92 });
anatofuz
parents:
diff changeset
93 } else {
anatofuz
parents:
diff changeset
94 sendMessage(llvm::json::Object{
anatofuz
parents:
diff changeset
95 {"jsonrpc", "2.0"},
anatofuz
parents:
diff changeset
96 {"id", std::move(ID)},
anatofuz
parents:
diff changeset
97 {"error", encodeError(Result.takeError())},
anatofuz
parents:
diff changeset
98 });
anatofuz
parents:
diff changeset
99 }
anatofuz
parents:
diff changeset
100 }
anatofuz
parents:
diff changeset
101
anatofuz
parents:
diff changeset
102 llvm::Error loop(MessageHandler &Handler) override {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
103 std::string JSON; // Messages may be large, reuse same big buffer.
150
anatofuz
parents:
diff changeset
104 while (!feof(In)) {
anatofuz
parents:
diff changeset
105 if (shutdownRequested())
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
106 return error(std::make_error_code(std::errc::operation_canceled),
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
107 "Got signal, shutting down");
150
anatofuz
parents:
diff changeset
108 if (ferror(In))
anatofuz
parents:
diff changeset
109 return llvm::errorCodeToError(
anatofuz
parents:
diff changeset
110 std::error_code(errno, std::system_category()));
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
111 if (readRawMessage(JSON)) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
112 if (auto Doc = llvm::json::parse(JSON)) {
150
anatofuz
parents:
diff changeset
113 vlog(Pretty ? "<<< {0:2}\n" : "<<< {0}\n", *Doc);
anatofuz
parents:
diff changeset
114 if (!handleMessage(std::move(*Doc), Handler))
anatofuz
parents:
diff changeset
115 return llvm::Error::success(); // we saw the "exit" notification.
anatofuz
parents:
diff changeset
116 } else {
anatofuz
parents:
diff changeset
117 // Parse error. Log the raw message.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
118 vlog("<<< {0}\n", JSON);
150
anatofuz
parents:
diff changeset
119 elog("JSON parse error: {0}", llvm::toString(Doc.takeError()));
anatofuz
parents:
diff changeset
120 }
anatofuz
parents:
diff changeset
121 }
anatofuz
parents:
diff changeset
122 }
anatofuz
parents:
diff changeset
123 return llvm::errorCodeToError(std::make_error_code(std::errc::io_error));
anatofuz
parents:
diff changeset
124 }
anatofuz
parents:
diff changeset
125
anatofuz
parents:
diff changeset
126 private:
anatofuz
parents:
diff changeset
127 // Dispatches incoming message to Handler onNotify/onCall/onReply.
anatofuz
parents:
diff changeset
128 bool handleMessage(llvm::json::Value Message, MessageHandler &Handler);
anatofuz
parents:
diff changeset
129 // Writes outgoing message to Out stream.
anatofuz
parents:
diff changeset
130 void sendMessage(llvm::json::Value Message) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
131 OutputBuffer.clear();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
132 llvm::raw_svector_ostream OS(OutputBuffer);
150
anatofuz
parents:
diff changeset
133 OS << llvm::formatv(Pretty ? "{0:2}" : "{0}", Message);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
134 Out << "Content-Length: " << OutputBuffer.size() << "\r\n\r\n"
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
135 << OutputBuffer;
150
anatofuz
parents:
diff changeset
136 Out.flush();
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
137 vlog(">>> {0}\n", OutputBuffer);
150
anatofuz
parents:
diff changeset
138 }
anatofuz
parents:
diff changeset
139
anatofuz
parents:
diff changeset
140 // Read raw string messages from input stream.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
141 bool readRawMessage(std::string &JSON) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
142 return Style == JSONStreamStyle::Delimited ? readDelimitedMessage(JSON)
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
143 : readStandardMessage(JSON);
150
anatofuz
parents:
diff changeset
144 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
145 bool readDelimitedMessage(std::string &JSON);
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
146 bool readStandardMessage(std::string &JSON);
150
anatofuz
parents:
diff changeset
147
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
148 llvm::SmallVector<char, 0> OutputBuffer;
150
anatofuz
parents:
diff changeset
149 std::FILE *In;
anatofuz
parents:
diff changeset
150 llvm::raw_ostream &Out;
anatofuz
parents:
diff changeset
151 llvm::raw_ostream &InMirror;
anatofuz
parents:
diff changeset
152 bool Pretty;
anatofuz
parents:
diff changeset
153 JSONStreamStyle Style;
anatofuz
parents:
diff changeset
154 };
anatofuz
parents:
diff changeset
155
anatofuz
parents:
diff changeset
156 bool JSONTransport::handleMessage(llvm::json::Value Message,
anatofuz
parents:
diff changeset
157 MessageHandler &Handler) {
anatofuz
parents:
diff changeset
158 // Message must be an object with "jsonrpc":"2.0".
anatofuz
parents:
diff changeset
159 auto *Object = Message.getAsObject();
anatofuz
parents:
diff changeset
160 if (!Object ||
anatofuz
parents:
diff changeset
161 Object->getString("jsonrpc") != llvm::Optional<llvm::StringRef>("2.0")) {
anatofuz
parents:
diff changeset
162 elog("Not a JSON-RPC 2.0 message: {0:2}", Message);
anatofuz
parents:
diff changeset
163 return false;
anatofuz
parents:
diff changeset
164 }
anatofuz
parents:
diff changeset
165 // ID may be any JSON value. If absent, this is a notification.
anatofuz
parents:
diff changeset
166 llvm::Optional<llvm::json::Value> ID;
anatofuz
parents:
diff changeset
167 if (auto *I = Object->get("id"))
anatofuz
parents:
diff changeset
168 ID = std::move(*I);
anatofuz
parents:
diff changeset
169 auto Method = Object->getString("method");
anatofuz
parents:
diff changeset
170 if (!Method) { // This is a response.
anatofuz
parents:
diff changeset
171 if (!ID) {
anatofuz
parents:
diff changeset
172 elog("No method and no response ID: {0:2}", Message);
anatofuz
parents:
diff changeset
173 return false;
anatofuz
parents:
diff changeset
174 }
anatofuz
parents:
diff changeset
175 if (auto *Err = Object->getObject("error"))
anatofuz
parents:
diff changeset
176 return Handler.onReply(std::move(*ID), decodeError(*Err));
anatofuz
parents:
diff changeset
177 // Result should be given, use null if not.
anatofuz
parents:
diff changeset
178 llvm::json::Value Result = nullptr;
anatofuz
parents:
diff changeset
179 if (auto *R = Object->get("result"))
anatofuz
parents:
diff changeset
180 Result = std::move(*R);
anatofuz
parents:
diff changeset
181 return Handler.onReply(std::move(*ID), std::move(Result));
anatofuz
parents:
diff changeset
182 }
anatofuz
parents:
diff changeset
183 // Params should be given, use null if not.
anatofuz
parents:
diff changeset
184 llvm::json::Value Params = nullptr;
anatofuz
parents:
diff changeset
185 if (auto *P = Object->get("params"))
anatofuz
parents:
diff changeset
186 Params = std::move(*P);
anatofuz
parents:
diff changeset
187
anatofuz
parents:
diff changeset
188 if (ID)
anatofuz
parents:
diff changeset
189 return Handler.onCall(*Method, std::move(Params), std::move(*ID));
anatofuz
parents:
diff changeset
190 else
anatofuz
parents:
diff changeset
191 return Handler.onNotify(*Method, std::move(Params));
anatofuz
parents:
diff changeset
192 }
anatofuz
parents:
diff changeset
193
anatofuz
parents:
diff changeset
194 // Tries to read a line up to and including \n.
anatofuz
parents:
diff changeset
195 // If failing, feof(), ferror(), or shutdownRequested() will be set.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
196 bool readLine(std::FILE *In, llvm::SmallVectorImpl<char> &Out) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
197 // Big enough to hold any reasonable header line. May not fit content lines
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
198 // in delimited mode, but performance doesn't matter for that mode.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
199 static constexpr int BufSize = 128;
150
anatofuz
parents:
diff changeset
200 size_t Size = 0;
anatofuz
parents:
diff changeset
201 Out.clear();
anatofuz
parents:
diff changeset
202 for (;;) {
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
203 Out.resize_for_overwrite(Size + BufSize);
150
anatofuz
parents:
diff changeset
204 // Handle EINTR which is sent when a debugger attaches on some platforms.
anatofuz
parents:
diff changeset
205 if (!retryAfterSignalUnlessShutdown(
anatofuz
parents:
diff changeset
206 nullptr, [&] { return std::fgets(&Out[Size], BufSize, In); }))
anatofuz
parents:
diff changeset
207 return false;
anatofuz
parents:
diff changeset
208 clearerr(In);
anatofuz
parents:
diff changeset
209 // If the line contained null bytes, anything after it (including \n) will
anatofuz
parents:
diff changeset
210 // be ignored. Fortunately this is not a legal header or JSON.
anatofuz
parents:
diff changeset
211 size_t Read = std::strlen(&Out[Size]);
anatofuz
parents:
diff changeset
212 if (Read > 0 && Out[Size + Read - 1] == '\n') {
anatofuz
parents:
diff changeset
213 Out.resize(Size + Read);
anatofuz
parents:
diff changeset
214 return true;
anatofuz
parents:
diff changeset
215 }
anatofuz
parents:
diff changeset
216 Size += Read;
anatofuz
parents:
diff changeset
217 }
anatofuz
parents:
diff changeset
218 }
anatofuz
parents:
diff changeset
219
anatofuz
parents:
diff changeset
220 // Returns None when:
anatofuz
parents:
diff changeset
221 // - ferror(), feof(), or shutdownRequested() are set.
anatofuz
parents:
diff changeset
222 // - Content-Length is missing or empty (protocol error)
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
223 bool JSONTransport::readStandardMessage(std::string &JSON) {
150
anatofuz
parents:
diff changeset
224 // A Language Server Protocol message starts with a set of HTTP headers,
anatofuz
parents:
diff changeset
225 // delimited by \r\n, and terminated by an empty line (\r\n).
anatofuz
parents:
diff changeset
226 unsigned long long ContentLength = 0;
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
227 llvm::SmallString<128> Line;
150
anatofuz
parents:
diff changeset
228 while (true) {
anatofuz
parents:
diff changeset
229 if (feof(In) || ferror(In) || !readLine(In, Line))
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
230 return false;
150
anatofuz
parents:
diff changeset
231 InMirror << Line;
anatofuz
parents:
diff changeset
232
anatofuz
parents:
diff changeset
233 llvm::StringRef LineRef(Line);
anatofuz
parents:
diff changeset
234
anatofuz
parents:
diff changeset
235 // We allow comments in headers. Technically this isn't part
anatofuz
parents:
diff changeset
236
anatofuz
parents:
diff changeset
237 // of the LSP specification, but makes writing tests easier.
anatofuz
parents:
diff changeset
238 if (LineRef.startswith("#"))
anatofuz
parents:
diff changeset
239 continue;
anatofuz
parents:
diff changeset
240
anatofuz
parents:
diff changeset
241 // Content-Length is a mandatory header, and the only one we handle.
anatofuz
parents:
diff changeset
242 if (LineRef.consume_front("Content-Length: ")) {
anatofuz
parents:
diff changeset
243 if (ContentLength != 0) {
anatofuz
parents:
diff changeset
244 elog("Warning: Duplicate Content-Length header received. "
anatofuz
parents:
diff changeset
245 "The previous value for this message ({0}) was ignored.",
anatofuz
parents:
diff changeset
246 ContentLength);
anatofuz
parents:
diff changeset
247 }
anatofuz
parents:
diff changeset
248 llvm::getAsUnsignedInteger(LineRef.trim(), 0, ContentLength);
anatofuz
parents:
diff changeset
249 continue;
anatofuz
parents:
diff changeset
250 } else if (!LineRef.trim().empty()) {
anatofuz
parents:
diff changeset
251 // It's another header, ignore it.
anatofuz
parents:
diff changeset
252 continue;
anatofuz
parents:
diff changeset
253 } else {
anatofuz
parents:
diff changeset
254 // An empty line indicates the end of headers.
anatofuz
parents:
diff changeset
255 // Go ahead and read the JSON.
anatofuz
parents:
diff changeset
256 break;
anatofuz
parents:
diff changeset
257 }
anatofuz
parents:
diff changeset
258 }
anatofuz
parents:
diff changeset
259
anatofuz
parents:
diff changeset
260 // The fuzzer likes crashing us by sending "Content-Length: 9999999999999999"
anatofuz
parents:
diff changeset
261 if (ContentLength > 1 << 30) { // 1024M
anatofuz
parents:
diff changeset
262 elog("Refusing to read message with long Content-Length: {0}. "
anatofuz
parents:
diff changeset
263 "Expect protocol errors",
anatofuz
parents:
diff changeset
264 ContentLength);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
265 return false;
150
anatofuz
parents:
diff changeset
266 }
anatofuz
parents:
diff changeset
267 if (ContentLength == 0) {
anatofuz
parents:
diff changeset
268 log("Warning: Missing Content-Length header, or zero-length message.");
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
269 return false;
150
anatofuz
parents:
diff changeset
270 }
anatofuz
parents:
diff changeset
271
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
272 JSON.resize(ContentLength);
150
anatofuz
parents:
diff changeset
273 for (size_t Pos = 0, Read; Pos < ContentLength; Pos += Read) {
anatofuz
parents:
diff changeset
274 // Handle EINTR which is sent when a debugger attaches on some platforms.
anatofuz
parents:
diff changeset
275 Read = retryAfterSignalUnlessShutdown(0, [&]{
anatofuz
parents:
diff changeset
276 return std::fread(&JSON[Pos], 1, ContentLength - Pos, In);
anatofuz
parents:
diff changeset
277 });
anatofuz
parents:
diff changeset
278 if (Read == 0) {
anatofuz
parents:
diff changeset
279 elog("Input was aborted. Read only {0} bytes of expected {1}.", Pos,
anatofuz
parents:
diff changeset
280 ContentLength);
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
281 return false;
150
anatofuz
parents:
diff changeset
282 }
anatofuz
parents:
diff changeset
283 InMirror << llvm::StringRef(&JSON[Pos], Read);
anatofuz
parents:
diff changeset
284 clearerr(In); // If we're done, the error was transient. If we're not done,
anatofuz
parents:
diff changeset
285 // either it was transient or we'll see it again on retry.
anatofuz
parents:
diff changeset
286 Pos += Read;
anatofuz
parents:
diff changeset
287 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
288 return true;
150
anatofuz
parents:
diff changeset
289 }
anatofuz
parents:
diff changeset
290
anatofuz
parents:
diff changeset
291 // For lit tests we support a simplified syntax:
anatofuz
parents:
diff changeset
292 // - messages are delimited by '---' on a line by itself
anatofuz
parents:
diff changeset
293 // - lines starting with # are ignored.
anatofuz
parents:
diff changeset
294 // This is a testing path, so favor simplicity over performance here.
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
295 // When returning false: feof(), ferror(), or shutdownRequested() will be set.
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
296 bool JSONTransport::readDelimitedMessage(std::string &JSON) {
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
297 JSON.clear();
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
298 llvm::SmallString<128> Line;
150
anatofuz
parents:
diff changeset
299 while (readLine(In, Line)) {
anatofuz
parents:
diff changeset
300 InMirror << Line;
anatofuz
parents:
diff changeset
301 auto LineRef = llvm::StringRef(Line).trim();
anatofuz
parents:
diff changeset
302 if (LineRef.startswith("#")) // comment
anatofuz
parents:
diff changeset
303 continue;
anatofuz
parents:
diff changeset
304
anatofuz
parents:
diff changeset
305 // found a delimiter
anatofuz
parents:
diff changeset
306 if (LineRef.rtrim() == "---")
anatofuz
parents:
diff changeset
307 break;
anatofuz
parents:
diff changeset
308
anatofuz
parents:
diff changeset
309 JSON += Line;
anatofuz
parents:
diff changeset
310 }
anatofuz
parents:
diff changeset
311
anatofuz
parents:
diff changeset
312 if (shutdownRequested())
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
313 return false;
150
anatofuz
parents:
diff changeset
314 if (ferror(In)) {
anatofuz
parents:
diff changeset
315 elog("Input error while reading message!");
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
316 return false;
150
anatofuz
parents:
diff changeset
317 }
221
79ff65ed7e25 LLVM12 Original
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 173
diff changeset
318 return true; // Including at EOF
150
anatofuz
parents:
diff changeset
319 }
anatofuz
parents:
diff changeset
320
anatofuz
parents:
diff changeset
321 } // namespace
anatofuz
parents:
diff changeset
322
anatofuz
parents:
diff changeset
323 std::unique_ptr<Transport> newJSONTransport(std::FILE *In,
anatofuz
parents:
diff changeset
324 llvm::raw_ostream &Out,
anatofuz
parents:
diff changeset
325 llvm::raw_ostream *InMirror,
anatofuz
parents:
diff changeset
326 bool Pretty,
anatofuz
parents:
diff changeset
327 JSONStreamStyle Style) {
anatofuz
parents:
diff changeset
328 return std::make_unique<JSONTransport>(In, Out, InMirror, Pretty, Style);
anatofuz
parents:
diff changeset
329 }
anatofuz
parents:
diff changeset
330
anatofuz
parents:
diff changeset
331 } // namespace clangd
anatofuz
parents:
diff changeset
332 } // namespace clang