LLDB mainline
JSONTransport.h
Go to the documentation of this file.
1//===-- JSONTransport.h ---------------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// Transport layer for encoding and decoding JSON protocol messages.
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef LLDB_HOST_JSONTRANSPORT_H
14#define LLDB_HOST_JSONTRANSPORT_H
15
16#include "lldb/Host/MainLoop.h"
19#include "lldb/Utility/Status.h"
20#include "lldb/lldb-forward.h"
21#include "llvm/ADT/FunctionExtras.h"
22#include "llvm/ADT/StringExtras.h"
23#include "llvm/ADT/StringRef.h"
24#include "llvm/Support/Error.h"
25#include "llvm/Support/ErrorHandling.h"
26#include "llvm/Support/FormatVariadic.h"
27#include "llvm/Support/JSON.h"
28#include "llvm/Support/raw_ostream.h"
29#include <atomic>
30#include <functional>
31#include <mutex>
32#include <optional>
33#include <string>
34#include <system_error>
35#include <type_traits>
36#include <utility>
37#include <variant>
38#include <vector>
39#if __cplusplus >= 202002L
40#include <concepts>
41#endif
42
44
45/// An error to indicate that the transport reached EOF but there were still
46/// unhandled contents in the read buffer.
48 : public llvm::ErrorInfo<TransportUnhandledContentsError> {
49public:
50 static char ID;
51
52 explicit TransportUnhandledContentsError(std::string unhandled_contents);
53
54 void log(llvm::raw_ostream &OS) const override;
55 std::error_code convertToErrorCode() const override;
56
57 const std::string &getUnhandledContents() const {
59 }
60
61private:
63};
64
65/// An error to indicate that the parameters of a Req, Resp or Evt could not be
66/// deserialized.
67class InvalidParams : public llvm::ErrorInfo<InvalidParams> {
68public:
69 static char ID;
70
71 explicit InvalidParams(std::string method, std::string context)
72 : m_method(std::move(method)), m_context(std::move(context)) {}
73
74 void log(llvm::raw_ostream &OS) const override;
75 std::error_code convertToErrorCode() const override;
76
77private:
78 /// The JSONRPC remote method call.
79 std::string m_method;
80
81 /// Additional context from the parsing failure, e.g. "missing value at
82 /// (root)[1].str".
83 std::string m_context;
84};
85
86/// An error to indicate that no handler was registered for a given method.
87class MethodNotFound : public llvm::ErrorInfo<MethodNotFound> {
88public:
89 static char ID;
90
91 static constexpr int kErrorCode = -32601;
92
93 explicit MethodNotFound(std::string method) : m_method(std::move(method)) {}
94
95 void log(llvm::raw_ostream &OS) const override;
96 std::error_code convertToErrorCode() const override;
97
98private:
99 std::string m_method;
100};
101
102#if __cplusplus >= 202002L
103/// A ProtocolDescriptor details the types used in a JSONTransport for handling
104/// transport communication.
105template <typename T>
106concept ProtocolDescriptor = requires {
107 typename T::Id;
108 typename T::Req;
109 typename T::Resp;
110 typename T::Evt;
111};
112#endif
113
114/// A transport is responsible for maintaining the connection to a client
115/// application, and reading/writing structured messages to it.
116///
117/// JSONTransport have limited thread safety requirements:
118/// - Messages will not be sent concurrently.
119/// - Messages MAY be sent while Run() is reading, or its callback is active.
120///
121#if __cplusplus >= 202002L
122template <ProtocolDescriptor Proto>
123#else
124template <typename Proto>
125#endif
127public:
128 using Req = typename Proto::Req;
129 using Resp = typename Proto::Resp;
130 using Evt = typename Proto::Evt;
131 using Message = std::variant<Req, Resp, Evt>;
132
133 virtual ~JSONTransport() = default;
134
135 /// Sends an event, a message that does not require a response.
136 virtual llvm::Error Send(const Evt &) = 0;
137 /// Sends a request, a message that expects a response.
138 virtual llvm::Error Send(const Req &) = 0;
139 /// Sends a response to a specific request.
140 virtual llvm::Error Send(const Resp &) = 0;
141
142 /// Implemented to handle incoming messages. (See `RegisterMessageHandler()`
143 /// below).
145 public:
146 virtual ~MessageHandler() = default;
147 /// Called when an event is received.
148 virtual void Received(const Evt &) = 0;
149 /// Called when a request is received.
150 virtual void Received(const Req &) = 0;
151 /// Called when a response is received.
152 virtual void Received(const Resp &) = 0;
153
154 /// Called when an error occurs while reading from the transport.
155 ///
156 /// NOTE: This does *NOT* indicate that a specific request failed, but that
157 /// there was an error in the underlying transport.
158 virtual void OnError(llvm::Error) = 0;
159
160 /// Called on EOF or client disconnect.
161 virtual void OnClosed() = 0;
162 };
163
164 /// RegisterMessageHandler registers the Transport with the given MainLoop and
165 /// handles any incoming messages using the given MessageHandler.
166 ///
167 /// If an unexpected error occurs, the MainLoop will be terminated and a log
168 /// message will include additional information about the termination reason.
169 virtual llvm::Error RegisterMessageHandler(MessageHandler &handler) = 0;
170
171protected:
172 template <typename... Ts> inline auto Logv(const char *Fmt, Ts &&...Vals) {
173 Log(llvm::formatv(Fmt, std::forward<Ts>(Vals)...).str());
174 }
175 virtual void Log(llvm::StringRef message) = 0;
176};
177
178/// An IOTransport sends and receives messages using an IOObject.
179template <typename Proto> class IOTransport : public JSONTransport<Proto> {
180public:
183
185 : m_loop(loop), m_in(in), m_out(out) {}
186
187 llvm::Error Send(const typename Proto::Evt &evt) override {
188 return Write(evt);
189 }
190
191 llvm::Error Send(const typename Proto::Req &req) override {
192 return Write(req);
193 }
194
195 llvm::Error Send(const typename Proto::Resp &resp) override {
196 return Write(resp);
197 }
198
199 llvm::Error RegisterMessageHandler(MessageHandler &handler) override {
200 Status status;
201 m_read_handle = m_loop.RegisterReadObject(
202 m_in, [this, &handler](MainLoopBase &base) { OnRead(base, handler); },
203 status);
204 return status.takeError();
205 }
206
207 /// Public for testing purposes, otherwise this should be an implementation
208 /// detail.
209 static constexpr size_t kReadBufferSize = 1024;
210
211protected:
212 llvm::Error Write(const llvm::json::Value &message) {
213 this->Logv("<-- {0}", message);
214 std::string output = Encode(message);
215 size_t bytes_written = output.size();
216 return m_out->Write(output.data(), bytes_written).takeError();
217 }
218
219 virtual llvm::Expected<std::vector<std::string>> Parse() = 0;
220 virtual std::string Encode(const llvm::json::Value &message) = 0;
221
222 llvm::SmallString<kReadBufferSize> m_buffer;
223
224private:
225 void OnRead(MainLoopBase &loop, MessageHandler &handler) {
226 char buf[kReadBufferSize];
227 size_t num_bytes = sizeof(buf);
228 if (Status status = m_in->Read(buf, num_bytes); status.Fail()) {
229 handler.OnError(status.takeError());
230 return;
231 }
232
233 if (num_bytes)
234 m_buffer.append(llvm::StringRef(buf, num_bytes));
235
236 // If the buffer has contents, try parsing any pending messages.
237 if (!m_buffer.empty()) {
238 llvm::Expected<std::vector<std::string>> raw_messages = Parse();
239 if (llvm::Error error = raw_messages.takeError()) {
240 handler.OnError(std::move(error));
241 return;
242 }
243
244 for (const std::string &raw_message : *raw_messages) {
245 llvm::Expected<Message> message =
246 llvm::json::parse<Message>(raw_message);
247 if (!message) {
248 handler.OnError(message.takeError());
249 return;
250 }
251
252 std::visit([&handler](auto &&msg) { handler.Received(msg); }, *message);
253 }
254 }
255
256 // Check if we reached EOF.
257 if (num_bytes == 0) {
258 // EOF reached, but there may still be unhandled contents in the buffer.
259 if (!m_buffer.empty())
260 handler.OnError(llvm::make_error<TransportUnhandledContentsError>(
261 std::string(m_buffer.str())));
262 // Move the read handle to a local before notifying the handler. The
263 // handler may destroy this transport (e.g. by erasing it from a
264 // connection map), so accessing members after OnClosed() is unsafe.
265 auto read_handle = std::move(m_read_handle);
266 handler.OnClosed();
267 }
268 }
269
274};
275
276/// A transport class for JSON with a HTTP header.
277#if __cplusplus >= 202002L
278template <ProtocolDescriptor Proto>
279#else
280template <typename Proto>
281#endif
283public:
284 using IOTransport<Proto>::IOTransport;
285
286protected:
287 /// Encodes messages based on
288 /// https://microsoft.github.io/debug-adapter-protocol/overview#base-protocol
289 std::string Encode(const llvm::json::Value &message) override {
290 std::string output;
291 std::string raw_message = llvm::formatv("{0}", message).str();
292 llvm::raw_string_ostream OS(output);
294 << std::to_string(raw_message.size()) << kEndOfHeader << raw_message;
295 return output;
296 }
297
298 /// Parses messages based on
299 /// https://microsoft.github.io/debug-adapter-protocol/overview#base-protocol
300 llvm::Expected<std::vector<std::string>> Parse() override {
301 std::vector<std::string> messages;
302 llvm::StringRef buffer = this->m_buffer;
303 while (buffer.contains(kEndOfHeader)) {
304 auto [headers, rest] = buffer.split(kEndOfHeader);
305 size_t content_length = 0;
306 // HTTP Headers are formatted like `<field-name> ':' [<field-value>]`.
307 for (const llvm::StringRef &header :
308 llvm::split(headers, kHeaderSeparator)) {
309 auto [key, value] = header.split(kHeaderFieldSeparator);
310 // 'Content-Length' is the only meaningful key at the moment. Others
311 // are ignored.
312 if (!key.equals_insensitive(kHeaderContentLength))
313 continue;
314
315 value = value.trim();
316 if (!llvm::to_integer(value, content_length, 10)) {
317 // Clear the buffer to avoid re-parsing this malformed message.
318 this->m_buffer.clear();
319 return llvm::createStringError(std::errc::invalid_argument,
320 "invalid content length: %s",
321 value.str().c_str());
322 }
323 }
324
325 // Check if we have enough data.
326 if (content_length > rest.size())
327 break;
328
329 llvm::StringRef body = rest.take_front(content_length);
330 buffer = rest.drop_front(content_length);
331 messages.emplace_back(body.str());
332 this->Logv("--> {0}", body);
333 }
334
335 // Store the remainder of the buffer for the next read callback.
336 this->m_buffer = buffer.str();
337
338 return std::move(messages);
339 }
340
341 static constexpr llvm::StringLiteral kHeaderContentLength = "Content-Length";
342 static constexpr llvm::StringLiteral kHeaderFieldSeparator = ":";
343 static constexpr llvm::StringLiteral kHeaderSeparator = "\r\n";
344 static constexpr llvm::StringLiteral kEndOfHeader = "\r\n\r\n";
345};
346
347/// A transport class for JSON RPC.
348#if __cplusplus >= 202002L
349template <ProtocolDescriptor Proto>
350#else
351template <typename Proto>
352#endif
353class JSONRPCTransport : public IOTransport<Proto> {
354public:
355 using IOTransport<Proto>::IOTransport;
356
357protected:
358 std::string Encode(const llvm::json::Value &message) override {
359 return llvm::formatv("{0}{1}", message, kMessageSeparator).str();
360 }
361
362 llvm::Expected<std::vector<std::string>> Parse() override {
363 std::vector<std::string> messages;
364 llvm::StringRef buf = this->m_buffer;
365 while (buf.contains(kMessageSeparator)) {
366 auto [raw_json, rest] = buf.split(kMessageSeparator);
367 buf = rest;
368 messages.emplace_back(raw_json.str());
369 this->Logv("--> {0}", raw_json);
370 }
371
372 // Store the remainder of the buffer for the next read callback.
373 this->m_buffer = buf.str();
374
375 return messages;
376 }
377
378 static constexpr llvm::StringLiteral kMessageSeparator = "\n";
379};
380
381/// A handler for the response to an outgoing request.
382template <typename T>
383using Reply =
384 std::conditional_t<std::is_void_v<T>,
385 llvm::unique_function<void(llvm::Error)>,
386 llvm::unique_function<void(llvm::Expected<T>)>>;
387
388namespace detail {
389template <typename R, typename P> struct request_t final {
390 using type = llvm::unique_function<void(const P &, Reply<R>)>;
391};
392template <typename R> struct request_t<R, void> final {
393 using type = llvm::unique_function<void(Reply<R>)>;
394};
395template <typename P> struct event_t final {
396 using type = llvm::unique_function<void(const P &)>;
397};
398template <> struct event_t<void> final {
399 using type = llvm::unique_function<void()>;
400};
401} // namespace detail
402
403template <typename R, typename P>
405
406/// A function to send an outgoing event.
407template <typename P> using OutgoingEvent = typename detail::event_t<P>::type;
408
409#if __cplusplus >= 202002L
410/// This represents a protocol description that includes additional helpers
411/// for constructing requests, responses and events to work with `Binder`.
412template <typename T>
413concept BindingBuilder =
414 ProtocolDescriptor<T> &&
415 requires(T::Id id, T::Req req, T::Resp resp, T::Evt evt,
416 llvm::StringRef method, std::optional<llvm::json::Value> params,
417 std::optional<llvm::json::Value> result, llvm::Error err) {
418 /// For initializing the unique sequence identifier;
419 { T::InitialId() } -> std::same_as<typename T::Id>;
420 /// Incrementing the sequence identifier.
421 { id++ } -> std::same_as<typename T::Id>;
422
423 /// Constructing protocol types
424 /// @{
425 /// Construct a new request.
426 { T::Make(id, method, params) } -> std::same_as<typename T::Req>;
427 /// Construct a new error response.
428 { T::Make(req, std::move(err)) } -> std::same_as<typename T::Resp>;
429 /// Construct a new success response.
430 { T::Make(req, result) } -> std::same_as<typename T::Resp>;
431 /// Construct a new event.
432 { T::Make(method, params) } -> std::same_as<typename T::Evt>;
433 /// @}
434
435 /// Keys for associated types.
436 /// @{
437 /// Looking up in flight responses.
438 { T::KeyFor(resp) } -> std::same_as<typename T::Id>;
439 /// Extract method from request.
440 { T::KeyFor(req) } -> std::same_as<std::string>;
441 /// Extract method from event.
442 { T::KeyFor(evt) } -> std::same_as<std::string>;
443 /// @}
444
445 /// Extracting information from associated types.
446 /// @{
447 /// Extract parameters from a request.
448 { T::Extract(req) } -> std::same_as<std::optional<llvm::json::Value>>;
449 /// Extract result from a response.
450 { T::Extract(resp) } -> std::same_as<llvm::Expected<llvm::json::Value>>;
451 /// Extract parameters from an event.
452 { T::Extract(evt) } -> std::same_as<std::optional<llvm::json::Value>>;
453 /// @}
454 };
455#endif
456
457/// Binder collects a table of functions that handle calls.
458///
459/// The wrapper takes care of parsing/serializing responses.
460///
461/// This allows a JSONTransport to handle incoming and outgoing requests and
462/// events.
463///
464/// A bind of an incoming request to a lambda.
465/// \code{cpp}
466/// Binder binder{transport};
467/// binder.bind<int, vector<int>>("adder", [](const vector<int> &params) {
468/// int sum = 0;
469/// for (int v : params)
470/// sum += v;
471/// return sum;
472/// });
473/// \endcode
474///
475/// A bind of an outgoing request.
476/// \code{cpp}
477/// OutgoingRequest<int, vector<int>> call_add =
478/// binder.bind<int, vector<int>>("add");
479/// call_add({1,2,3}, [](Expected<int> result) {
480/// cout << *result << "\n";
481/// });
482/// \endcode
483#if __cplusplus >= 202002L
484template <BindingBuilder Proto>
485#else
486template <typename Proto>
487#endif
489 using Req = typename Proto::Req;
490 using Resp = typename Proto::Resp;
491 using Evt = typename Proto::Evt;
492 using Id = typename Proto::Id;
495
496public:
498
499 Binder(const Binder &) = delete;
500 Binder &operator=(const Binder &) = delete;
501
502 /// Bind a handler on transport disconnect.
503 template <typename Fn, typename... Args>
504 void OnDisconnect(Fn &&fn, Args &&...args);
505
506 /// Bind a handler on error when communicating with the transport.
507 template <typename Fn, typename... Args>
508 void OnError(Fn &&fn, Args &&...args);
509
510 /// Bind a handler for an incoming request.
511 /// e.g. `bind("peek", &ThisModule::peek, this);`.
512 /// Handler should be e.g. `Expected<PeekResult> peek(const PeekParams&);`
513 /// PeekParams must be JSON parsable and PeekResult must be serializable.
514 template <typename Result, typename Params, typename Fn, typename... Args>
515 void Bind(llvm::StringLiteral method, Fn &&fn, Args &&...args);
516
517 /// Bind a handler for an incoming event.
518 /// e.g. `bind("peek", &ThisModule::peek, this);`
519 /// Handler should be e.g. `void peek(const PeekParams&);`
520 /// PeekParams must be JSON parsable.
521 template <typename Params, typename Fn, typename... Args>
522 void Bind(llvm::StringLiteral method, Fn &&fn, Args &&...args);
523
524 /// Bind a function object to be used for outgoing requests.
525 /// e.g. `OutgoingRequest<Params, Result> Edit = bind("edit");`
526 /// Params must be JSON-serializable, Result must be parsable.
527 template <typename Result, typename Params>
528 OutgoingRequest<Result, Params> Bind(llvm::StringLiteral method);
529
530 /// Bind a function object to be used for outgoing events.
531 /// e.g. `OutgoingEvent<LogParams> Log = bind("log");`
532 /// LogParams must be JSON-serializable.
533 template <typename Params>
534 OutgoingEvent<Params> Bind(llvm::StringLiteral method);
535
536 void Received(const Evt &evt) override {
537 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
538 auto it = m_event_handlers.find(Proto::KeyFor(evt));
539 if (it == m_event_handlers.end()) {
540 OnError(llvm::createStringError(
541 llvm::formatv("no handled for event {0}", toJSON(evt))));
542 return;
543 }
544 it->second(evt);
545 }
546
547 void Received(const Req &req) override {
548 ReplyOnce reply(req, &m_transport, this);
549
550 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
551 auto it = m_request_handlers.find(Proto::KeyFor(req));
552 if (it == m_request_handlers.end()) {
553 reply(Proto::Make(req, llvm::createStringError("method not found")));
554 return;
555 }
556
557 it->second(req, std::move(reply));
558 }
559
560 void Received(const Resp &resp) override {
561 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
562
563 Id id = Proto::KeyFor(resp);
564 auto it = m_pending_responses.find(id);
565 if (it == m_pending_responses.end()) {
566 OnError(llvm::createStringError(
567 llvm::formatv("no pending request for {0}", toJSON(resp))));
568 return;
569 }
570
571 it->second(resp);
572 m_pending_responses.erase(it);
573 }
574
575 void OnError(llvm::Error err) override {
576 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
577 if (m_error_handler)
578 m_error_handler(std::move(err));
579 }
580
581 void OnClosed() override {
582 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
585 }
586
587private:
588 template <typename T>
589 llvm::Expected<T> static Parse(const llvm::json::Value &raw,
590 llvm::StringRef method);
591
592 template <typename T> using Callback = llvm::unique_function<T>;
593
594 std::recursive_mutex m_mutex;
597 std::map<Id, Callback<void(const Resp &)>> m_pending_responses;
598 llvm::StringMap<Callback<void(const Req &, Callback<void(const Resp &)>)>>
600 llvm::StringMap<Callback<void(const Evt &)>> m_event_handlers;
602 Callback<void(llvm::Error)> m_error_handler;
603
604 /// Function object to reply to a call.
605 /// Each instance must be called exactly once, otherwise:
606 /// - the bug is logged, and (in debug mode) an assert will fire
607 /// - if there was no reply, an error reply is sent
608 /// - if there were multiple replies, only the first is sent
609 class ReplyOnce {
610 std::atomic<bool> replied = {false};
611 const Req req;
612 Transport *transport; // Null when moved-from.
613 MessageHandler *handler; // Null when moved-from.
614
615 public:
621 : replied(other.replied.load()), req(other.req),
622 transport(other.transport), handler(other.handler) {
623 other.transport = nullptr;
624 other.handler = nullptr;
625 }
627 ReplyOnce(const ReplyOnce &) = delete;
628 ReplyOnce &operator=(const ReplyOnce &) = delete;
629
631 if (transport && handler && !replied) {
632 assert(false && "must reply to all calls!");
633 (*this)(Proto::Make(req, llvm::createStringError("failed to reply")));
634 }
635 }
636
637 void operator()(const Resp &resp) {
638 assert(transport && handler && "moved-from!");
639 if (replied.exchange(true)) {
640 assert(false && "must reply to each call only once!");
641 return;
642 }
643
644 if (llvm::Error error = transport->Send(resp))
645 handler->OnError(std::move(error));
646 }
647 };
648};
649
650#if __cplusplus >= 202002L
651template <BindingBuilder Proto>
652#else
653template <typename Proto>
654#endif
655template <typename Fn, typename... Args>
656void Binder<Proto>::OnDisconnect(Fn &&fn, Args &&...args) {
657 m_disconnect_handler = [fn, args...]() mutable {
658 std::invoke(std::forward<Fn>(fn), std::forward<Args>(args)...);
659 };
660}
661
662#if __cplusplus >= 202002L
663template <BindingBuilder Proto>
664#else
665template <typename Proto>
666#endif
667template <typename Fn, typename... Args>
668void Binder<Proto>::OnError(Fn &&fn, Args &&...args) {
669 m_error_handler = [fn, args...](llvm::Error error) mutable {
670 std::invoke(std::forward<Fn>(fn), std::forward<Args>(args)...,
671 std::move(error));
672 };
673}
674
675#if __cplusplus >= 202002L
676template <BindingBuilder Proto>
677#else
678template <typename Proto>
679#endif
680template <typename Result, typename Params, typename Fn, typename... Args>
681void Binder<Proto>::Bind(llvm::StringLiteral method, Fn &&fn, Args &&...args) {
682 assert(m_request_handlers.find(method) == m_request_handlers.end() &&
683 "request already bound");
684 if constexpr (std::is_void_v<Result> && std::is_void_v<Params>) {
685 m_request_handlers[method] =
686 [fn, args...](const Req &req,
687 llvm::unique_function<void(const Resp &)> reply) mutable {
688 llvm::Error result =
689 std::invoke(std::forward<Fn>(fn), std::forward<Args>(args)...);
690 reply(Proto::Make(req, std::move(result)));
691 };
692 } else if constexpr (std::is_void_v<Params>) {
693 m_request_handlers[method] =
694 [fn, args...](const Req &req,
695 llvm::unique_function<void(const Resp &)> reply) mutable {
696 llvm::Expected<Result> result =
697 std::invoke(std::forward<Fn>(fn), std::forward<Args>(args)...);
698 if (!result)
699 return reply(Proto::Make(req, result.takeError()));
700 reply(Proto::Make(req, toJSON(*result)));
701 };
702 } else if constexpr (std::is_void_v<Result>) {
703 m_request_handlers[method] =
704 [method, fn,
705 args...](const Req &req,
706 llvm::unique_function<void(const Resp &)> reply) mutable {
707 llvm::Expected<Params> params =
708 Parse<Params>(Proto::Extract(req), method);
709 if (!params)
710 return reply(Proto::Make(req, params.takeError()));
711
712 llvm::Error result = std::invoke(
713 std::forward<Fn>(fn), std::forward<Args>(args)..., *params);
714 reply(Proto::Make(req, std::move(result)));
715 };
716 } else {
717 m_request_handlers[method] =
718 [method, fn,
719 args...](const Req &req,
720 llvm::unique_function<void(const Resp &)> reply) mutable {
721 llvm::Expected<Params> params =
722 Parse<Params>(Proto::Extract(req), method);
723 if (!params)
724 return reply(Proto::Make(req, params.takeError()));
725
726 llvm::Expected<Result> result = std::invoke(
727 std::forward<Fn>(fn), std::forward<Args>(args)..., *params);
728 if (!result)
729 return reply(Proto::Make(req, result.takeError()));
730
731 reply(Proto::Make(req, toJSON(*result)));
732 };
733 }
734}
735
736#if __cplusplus >= 202002L
737template <BindingBuilder Proto>
738#else
739template <typename Proto>
740#endif
741template <typename Params, typename Fn, typename... Args>
742void Binder<Proto>::Bind(llvm::StringLiteral method, Fn &&fn, Args &&...args) {
743 assert(m_event_handlers.find(method) == m_event_handlers.end() &&
744 "event already bound");
745 if constexpr (std::is_void_v<Params>) {
746 m_event_handlers[method] = [fn, args...](const Evt &) mutable {
747 std::invoke(std::forward<Fn>(fn), std::forward<Args>(args)...);
748 };
749 } else {
750 m_event_handlers[method] = [this, method, fn,
751 args...](const Evt &evt) mutable {
752 llvm::Expected<Params> params =
753 Parse<Params>(Proto::Extract(evt), method);
754 if (!params)
755 return OnError(params.takeError());
756 std::invoke(std::forward<Fn>(fn), std::forward<Args>(args)..., *params);
757 };
758 }
759}
760
761#if __cplusplus >= 202002L
762template <BindingBuilder Proto>
763#else
764template <typename Proto>
765#endif
766template <typename Result, typename Params>
768Binder<Proto>::Bind(llvm::StringLiteral method) {
769 if constexpr (std::is_void_v<Result> && std::is_void_v<Params>) {
770 return [this, method](Reply<Result> fn) {
771 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
772 Id id = ++m_seq;
773 Req req = Proto::Make(id, method, std::nullopt);
774 m_pending_responses[id] = [fn = std::move(fn)](const Resp &resp) mutable {
775 llvm::Expected<llvm::json::Value> result = Proto::Extract(resp);
776 if (!result)
777 return fn(result.takeError());
778 fn(llvm::Error::success());
779 };
780 if (llvm::Error error = m_transport.Send(req))
781 OnError(std::move(error));
782 };
783 } else if constexpr (std::is_void_v<Params>) {
784 return [this, method](Reply<Result> fn) {
785 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
786 Id id = ++m_seq;
787 Req req = Proto::Make(id, method, std::nullopt);
788 m_pending_responses[id] = [fn = std::move(fn),
789 method](const Resp &resp) mutable {
790 llvm::Expected<llvm::json::Value> result = Proto::Extract(resp);
791 if (!result)
792 return fn(result.takeError());
793 fn(Parse<Result>(*result, method));
794 };
795 if (llvm::Error error = m_transport.Send(req))
796 OnError(std::move(error));
797 };
798 } else if constexpr (std::is_void_v<Result>) {
799 return [this, method](const Params &params, Reply<Result> fn) {
800 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
801 Id id = ++m_seq;
802 Req req = Proto::Make(id, method, llvm::json::Value(params));
803 m_pending_responses[id] = [fn = std::move(fn)](const Resp &resp) mutable {
804 llvm::Expected<llvm::json::Value> result = Proto::Extract(resp);
805 if (!result)
806 return fn(result.takeError());
807 fn(llvm::Error::success());
808 };
809 if (llvm::Error error = m_transport.Send(req))
810 OnError(std::move(error));
811 };
812 } else {
813 return [this, method](const Params &params, Reply<Result> fn) {
814 std::scoped_lock<std::recursive_mutex> guard(m_mutex);
815 Id id = ++m_seq;
816 Req req = Proto::Make(id, method, llvm::json::Value(params));
817 m_pending_responses[id] = [fn = std::move(fn),
818 method](const Resp &resp) mutable {
819 llvm::Expected<llvm::json::Value> result = Proto::Extract(resp);
820 if (llvm::Error err = result.takeError())
821 return fn(std::move(err));
822 fn(Parse<Result>(*result, method));
823 };
824 if (llvm::Error error = m_transport.Send(req))
825 OnError(std::move(error));
826 };
827 }
828}
829
830#if __cplusplus >= 202002L
831template <BindingBuilder Proto>
832#else
833template <typename Proto>
834#endif
835template <typename Params>
836OutgoingEvent<Params> Binder<Proto>::Bind(llvm::StringLiteral method) {
837 if constexpr (std::is_void_v<Params>) {
838 return [this, method]() {
839 if (llvm::Error error =
840 m_transport.Send(Proto::Make(method, std::nullopt)))
841 OnError(std::move(error));
842 };
843 } else {
844 return [this, method](const Params &params) {
845 if (llvm::Error error =
846 m_transport.Send(Proto::Make(method, toJSON(params))))
847 OnError(std::move(error));
848 };
849 }
850}
851
852#if __cplusplus >= 202002L
853template <BindingBuilder Proto>
854#else
855template <typename Proto>
856#endif
857template <typename T>
858llvm::Expected<T> Binder<Proto>::Parse(const llvm::json::Value &raw,
859 llvm::StringRef method) {
860 T result;
861 llvm::json::Path::Root root;
862 if (!fromJSON(raw, result, root)) {
863 // Dump the relevant parts of the broken message.
864 std::string context;
865 llvm::raw_string_ostream OS(context);
866 root.printErrorContext(raw, OS);
867 return llvm::make_error<InvalidParams>(method.str(), context);
868 }
869 return std::move(result);
870}
871
872} // namespace lldb_private::transport
873
874#endif
static llvm::raw_ostream & error(Stream &strm)
A command line argument class.
Definition Args.h:33
std::unique_ptr< ReadHandle > ReadHandleUP
An error handling class.
Definition Status.h:118
llvm::Error takeError()
Definition Status.h:170
ReplyOnce & operator=(const ReplyOnce &)=delete
ReplyOnce(const Req req, Transport *transport, MessageHandler *handler)
ReplyOnce & operator=(ReplyOnce &&)=delete
void Received(const Resp &resp) override
Called when a response is received.
JSONTransport< Proto > Transport
typename Transport::MessageHandler MessageHandler
void OnDisconnect(Fn &&fn, Args &&...args)
Bind a handler on transport disconnect.
void Bind(llvm::StringLiteral method, Fn &&fn, Args &&...args)
Bind a handler for an incoming request.
void Bind(llvm::StringLiteral method, Fn &&fn, Args &&...args)
Bind a handler for an incoming event.
llvm::StringMap< Callback< void(const Req &, Callback< void(const Resp &)>)> > m_request_handlers
void OnClosed() override
Called on EOF or client disconnect.
void OnError(Fn &&fn, Args &&...args)
Bind a handler on error when communicating with the transport.
void Received(const Evt &evt) override
Called when an event is received.
OutgoingRequest< Result, Params > Bind(llvm::StringLiteral method)
Bind a function object to be used for outgoing requests.
std::map< Id, Callback< void(const Resp &)> > m_pending_responses
Binder & operator=(const Binder &)=delete
Binder(Transport &transport)
void Received(const Req &req) override
Called when a request is received.
static llvm::Expected< T > Parse(const llvm::json::Value &raw, llvm::StringRef method)
OutgoingEvent< Params > Bind(llvm::StringLiteral method)
Bind a function object to be used for outgoing events.
Binder(const Binder &)=delete
void OnError(llvm::Error err) override
Called when an error occurs while reading from the transport.
llvm::unique_function< T > Callback
llvm::StringMap< Callback< void(const Evt &)> > m_event_handlers
A transport class for JSON with a HTTP header.
static constexpr llvm::StringLiteral kHeaderFieldSeparator
static constexpr llvm::StringLiteral kEndOfHeader
static constexpr llvm::StringLiteral kHeaderSeparator
std::string Encode(const llvm::json::Value &message) override
Encodes messages based on https://microsoft.github.io/debug-adapter-protocol/overview#base-protocol.
llvm::Expected< std::vector< std::string > > Parse() override
Parses messages based on https://microsoft.github.io/debug-adapter-protocol/overview#base-protocol.
static constexpr llvm::StringLiteral kHeaderContentLength
void OnRead(MainLoopBase &loop, MessageHandler &handler)
IOTransport(MainLoop &loop, lldb::IOObjectSP in, lldb::IOObjectSP out)
static constexpr size_t kReadBufferSize
Public for testing purposes, otherwise this should be an implementation detail.
typename JSONTransport< Proto >::MessageHandler MessageHandler
llvm::Error Send(const typename Proto::Resp &resp) override
typename JSONTransport< Proto >::Message Message
llvm::Error Send(const typename Proto::Evt &evt) override
virtual std::string Encode(const llvm::json::Value &message)=0
virtual llvm::Expected< std::vector< std::string > > Parse()=0
MainLoop::ReadHandleUP m_read_handle
llvm::Error Write(const llvm::json::Value &message)
llvm::Error Send(const typename Proto::Req &req) override
llvm::SmallString< kReadBufferSize > m_buffer
llvm::Error RegisterMessageHandler(MessageHandler &handler) override
RegisterMessageHandler registers the Transport with the given MainLoop and handles any incoming messa...
std::error_code convertToErrorCode() const override
std::string m_context
Additional context from the parsing failure, e.g.
InvalidParams(std::string method, std::string context)
std::string m_method
The JSONRPC remote method call.
void log(llvm::raw_ostream &OS) const override
A transport class for JSON RPC.
llvm::Expected< std::vector< std::string > > Parse() override
static constexpr llvm::StringLiteral kMessageSeparator
std::string Encode(const llvm::json::Value &message) override
Implemented to handle incoming messages.
virtual void OnError(llvm::Error)=0
Called when an error occurs while reading from the transport.
virtual void OnClosed()=0
Called on EOF or client disconnect.
virtual void Received(const Req &)=0
Called when a request is received.
virtual void Received(const Evt &)=0
Called when an event is received.
virtual void Received(const Resp &)=0
Called when a response is received.
A transport is responsible for maintaining the connection to a client application,...
virtual llvm::Error Send(const Resp &)=0
Sends a response to a specific request.
std::variant< Req, Resp, Evt > Message
virtual llvm::Error Send(const Evt &)=0
Sends an event, a message that does not require a response.
virtual llvm::Error RegisterMessageHandler(MessageHandler &handler)=0
RegisterMessageHandler registers the Transport with the given MainLoop and handles any incoming messa...
auto Logv(const char *Fmt, Ts &&...Vals)
virtual llvm::Error Send(const Req &)=0
Sends a request, a message that expects a response.
virtual void Log(llvm::StringRef message)=0
std::error_code convertToErrorCode() const override
void log(llvm::raw_ostream &OS) const override
void log(llvm::raw_ostream &OS) const override
TransportUnhandledContentsError(std::string unhandled_contents)
std::conditional_t< std::is_void_v< T >, llvm::unique_function< void(llvm::Error)>, llvm::unique_function< void(llvm::Expected< T >)> > Reply
A handler for the response to an outgoing request.
typename detail::event_t< P >::type OutgoingEvent
A function to send an outgoing event.
typename detail::request_t< R, P >::type OutgoingRequest
bool fromJSON(const llvm::json::Value &value, AcceleratorActions &data, llvm::json::Path path)
llvm::json::Value toJSON(const AcceleratorActions &data)
MainLoopPosix MainLoop
Definition MainLoop.h:20
std::shared_ptr< lldb_private::IOObject > IOObjectSP
llvm::unique_function< void(const P &)> type
llvm::unique_function< void(Reply< R >)> type
llvm::unique_function< void(const P &, Reply< R >)> type