JSON RPC
JSON-RPC 2.0 library for C++
exception.h
Go to the documentation of this file.
1#ifndef CC75354D_5C03_4B34_B773_96A9E6189611
2#define CC75354D_5C03_4B34_B773_96A9E6189611
3
4/**
5 * @file exception.h
6 * @brief Contains the definition of the JSON RPC Exception class.
7 * @see https://www.jsonrpc.org/specification#error_object
8 */
9
10#include <exception>
11#include <string>
12#include <string_view>
13#include <nlohmann/json.hpp>
14
15#include "export.h"
16
17namespace wwa::json_rpc {
18
19/**
20 * @defgroup error_message Error Messages
21 * @brief Error messages used by the library. These constants can be useful in unit tests.
22 * @{
23 */
24
25/**
26 * @brief Error message for when the request is not a JSON-RPC 2.0 request.
27 * @see exception::INVALID_REQUEST
28 * @see https://www.jsonrpc.org/specification#request_object
29 * @details > `jsonrpc`: A String specifying the version of the JSON-RPC protocol. MUST be exactly "2.0".
30 */
31static constexpr std::string_view err_not_jsonrpc_2_0_request = "Not a JSON-RPC 2.0 request";
32
33/**
34 * @brief Error request for when the request is not valid.
35 * @see exception::INVALID_REQUEST
36 * @see https://www.jsonrpc.org/specification#request_object
37 */
38static constexpr std::string_view err_bad_request = "Bad request";
39
40/**
41 * @brief Error message for when the parameters passed to the method are not correct.
42 * @see exception::INVALID_PARAMS
43 */
44static constexpr std::string_view err_invalid_params_passed_to_method = "Invalid parameters passed to method";
45
46/**
47 * @brief Error message for when the method is not found.
48 * @see exception::METHOD_NOT_FOUND
49 */
50static constexpr std::string_view err_method_not_found = "Method not found";
51
52/**
53 * @brief Error message for when the method is empty.
54 * @see exception::INVALID_REQUEST
55 */
56static constexpr std::string_view err_empty_method = "Method cannot be empty";
57
58/**
59 * @brief Error message for when the parameters are not an array or an object.
60 * @see exception::INVALID_PARAMS
61 */
62static constexpr std::string_view err_bad_params_type = "Parameters must be either an array or an object or omitted";
63
64/**
65 * @brief Error message for when the ID is not a number, a string, or null.
66 * @see exception::INVALID_REQUEST
67 * @see https://www.jsonrpc.org/specification#request_object
68 * @details > An identifier established by the Client that MUST contain a String, Number, or NULL value if included.
69 */
70static constexpr std::string_view err_bad_id_type = "ID must be either a number, a string, or null";
71
72/**
73 * @brief Error message for when the batch request is empty.
74 * @see exception::INVALID_REQUEST
75 * @see https://www.jsonrpc.org/specification#batch
76 */
77static constexpr std::string_view err_empty_batch = "Empty batch request";
78/** @} */
79
80/**
81 * @brief JSON RPC Exception class.
82 *
83 * This class represents an exception that can occur during the processing of JSON RPC requests.
84 * It includes an error code, a message, and optional additional data.
85 */
86class WWA_JSONRPC_EXPORT exception : public std::exception {
87public:
88 /**
89 * @defgroup error_codes Error Codes
90 * @brief Error codes defined by the [JSON-RPC 2.0 specification](https://www.jsonrpc.org/specification#request_object).
91 * @see https://www.jsonrpc.org/specification#error_object
92 * @{
93 */
94
95 /**
96 * @brief Invalid JSON was received by the server.
97 *
98 * An error occurred on the server while parsing the JSON text.
99 */
100 static constexpr int PARSE_ERROR = -32700;
101
102 /**
103 * @brief The JSON sent is not a valid Request object.
104 */
105 static constexpr int INVALID_REQUEST = -32600;
106
107 /**
108 * @brief The method does not exist or is not available.
109 */
110 static constexpr int METHOD_NOT_FOUND = -32601;
111
112 /**
113 * @brief Invalid method parameter(s).
114 */
115 static constexpr int INVALID_PARAMS = -32602;
116
117 /**
118 * @brief Internal JSON-RPC error.
119 */
120 static constexpr int INTERNAL_ERROR = -32603;
121 /** @} */
122
123 /**
124 * @brief Construct a new exception object with additional data.
125 *
126 * @tparam T Type of the @a data. Must be [convertible to `nlohmann::json`](https://github.com/nlohmann/json?tab=readme-ov-file#arbitrary-types-conversions).
127 * @param code Indicates the error type that occurred.
128 * @param message Provides a short description of the error. The message SHOULD be limited to a concise single sentence.
129 * @param data Additional information about the error.
130 *
131 * @see https://www.jsonrpc.org/specification#error_object
132 */
133 template<typename T>
134 exception(int code, std::string_view message, const T& data) : m_message(message), m_data(data), m_code(code)
135 {}
136
137 /**
138 * @brief Construct a new exception object.
139 *
140 * @param code Indicates the error type that occurred.
141 * @param message Provides a short description of the error. The message SHOULD be limited to a concise single sentence.
142 *
143 * @see https://www.jsonrpc.org/specification#error_object
144 */
145 exception(int code, std::string_view message) : m_message(message), m_code(code) {}
146
147 /**
148 * @brief Default copy constructor.
149 */
150 exception(const exception&) = default;
151
152 /**
153 * @brief Default move constructor.
154 */
155 exception(exception&&) = default;
156
157 /**
158 * @brief Default copy assignment operator.
159 *
160 * @param rhs Right-hand side of the assignment.
161 * @return Reference to this object.
162 */
163 exception& operator=(const exception& rhs) = default;
164
165 /**
166 * @brief Default move assignment operator.
167 *
168 * @param rhs Right-hand side of the assignment.
169 * @return Reference to this object.
170 */
171 exception& operator=(exception&& rhs) = default;
172
173 /**
174 * @brief Default destructor
175 */
176 ~exception() override;
177
178 /**
179 * @brief Returns the error code.
180 *
181 * @return Error code.
182 */
183 [[nodiscard]] int code() const noexcept { return this->m_code; }
184
185 /**
186 * @brief Returns the error message.
187 *
188 * @return Error message.
189 */
190 [[nodiscard]] const std::string& message() const noexcept { return this->m_message; }
191
192 /**
193 * @brief Returns custom data associated with the error.
194 *
195 * @return Custom data in JSON format.
196 */
197 [[nodiscard]] const nlohmann::json& data() const noexcept { return this->m_data; }
198
199 /**
200 * @brief Returns the error message.
201 *
202 * @see https://en.cppreference.com/w/cpp/error/exception/what
203 * @return Pointer to a null-terminated string with explanatory information.
204 *
205 * @see message()
206 */
207 [[nodiscard]] const char* what() const noexcept override { return this->m_message.c_str(); }
208
209 /**
210 * @brief Returns the error message as an Error Object.
211 *
212 * @see https://www.jsonrpc.org/specification#error_object
213 * @return Error Object as JSON.
214 */
216 {
217 nlohmann::json j{
218 {"code", this->m_code},
219 {"message", this->m_message},
220 };
221
222 if (!this->m_data.is_null()) {
223 j["data"] = this->m_data;
224 }
225
226 return j;
227 }
228
229private:
230 std::string m_message; ///< Error message.
231 nlohmann::json m_data; ///< Custom data associated with the error.
232 int m_code; ///< Error code.
233};
234
235/**
236 * @brief Exception thrown when the method is not found.
237 */
251
252} // namespace wwa::json_rpc
253
254#endif /* CC75354D_5C03_4B34_B773_96A9E6189611 */
Private implementation of the JSON RPC dispatcher class.
static std::uint64_t get_and_increment_counter() noexcept
Generates a unique request ID.
A class that manages JSON RPC method handlers and processes JSON RPC requests.
Definition dispatcher.h:77
virtual nlohmann::json do_process_request(const nlohmann::json &request, const std::any &data, bool is_batch, std::uint64_t unique_id)
Processes a single, non-batch JSON RPC request.
dispatcher & operator=(dispatcher &&rhs)=default
Move assignment operator.
dispatcher(dispatcher &&rhs)=default
Move constructor.
void add_ex(std::string_view method, F &&f, C instance)
Adds a method handler with a context parameter and a class instance.
Definition dispatcher.h:286
constexpr auto create_closure(C inst, F &&f) const
Creates a closure for invoking a member function with JSON parameters.
Definition dispatcher.h:453
virtual void request_parsed(const jsonrpc_request &request, const std::any &data, std::uint64_t unique_id)
Invoked after the request has been parsed.
void add_internal_method(std::string_view method, handler_t &&handler)
Adds a method handler for the specified method.
nlohmann::json process_request(const nlohmann::json &request, const std::any &data={})
Processes a JSON RPC request.
virtual void request_failed(const nlohmann::json &request_id, const std::exception *e, bool is_batch, std::uint64_t unique_id)
Invoked when a request fails.
virtual nlohmann::json invoke(const std::string &method, const nlohmann::json &params, const dispatcher::context_t &ctx, std::uint64_t unique_id)
Invokes a method handler.
void add(std::string_view method, F &&f)
Adds a method handler f for the method method.
Definition dispatcher.h:203
void add(std::string_view method, F &&f, C instance)
Adds a method to the dispatcher with the specified instance and function.
Definition dispatcher.h:221
virtual nlohmann::json process_batch_request(const nlohmann::json &request, const std::any &data, std::uint64_t unique_id)
Processes a batch request.
dispatcher()
Class constructor.
dispatcher(const dispatcher &)=delete
dispatcher & operator=(const dispatcher &)=delete
virtual ~dispatcher()
Class destructor.
void add_ex(std::string_view method, F &&f)
Adds a method handler with a context parameter.
Definition dispatcher.h:263
std::unique_ptr< dispatcher_private > d_ptr
Pointer to the implementation (Pimpl idiom).
Definition dispatcher.h:411
JSON RPC Exception class.
Definition exception.h:86
exception(const exception &)=default
Default copy constructor.
~exception() override
Default destructor.
nlohmann::json m_data
Custom data associated with the error.
Definition exception.h:231
std::string m_message
Error message.
Definition exception.h:230
int code() const noexcept
Returns the error code.
Definition exception.h:183
const std::string & message() const noexcept
Returns the error message.
Definition exception.h:190
exception(exception &&)=default
Default move constructor.
exception(int code, std::string_view message)
Construct a new exception object.
Definition exception.h:145
exception & operator=(const exception &rhs)=default
Default copy assignment operator.
const nlohmann::json & data() const noexcept
Returns custom data associated with the error.
Definition exception.h:197
exception & operator=(exception &&rhs)=default
Default move assignment operator.
const char * what() const noexcept override
Returns the error message.
Definition exception.h:207
nlohmann::json to_json() const
Returns the error message as an Error Object.
Definition exception.h:215
exception(int code, std::string_view message, const T &data)
Construct a new exception object with additional data.
Definition exception.h:134
int m_code
Error code.
Definition exception.h:232
Exception thrown when the method is not found.
Definition exception.h:238
method_not_found_exception(method_not_found_exception &&)=default
method_not_found_exception & operator=(method_not_found_exception &&)=default
method_not_found_exception & operator=(const method_not_found_exception &)=default
~method_not_found_exception() override
Default destructor.
method_not_found_exception(const method_not_found_exception &)=default
#define WWA_JSONRPC_EXPORT
Macro for exporting symbols when building the library dynamically or importing symbols when using the...
Definition export.h:42
static constexpr int INVALID_PARAMS
Invalid method parameter(s).
Definition exception.h:115
static constexpr int INTERNAL_ERROR
Internal JSON-RPC error.
Definition exception.h:120
static constexpr int METHOD_NOT_FOUND
The method does not exist or is not available.
Definition exception.h:110
static constexpr int INVALID_REQUEST
The JSON sent is not a valid Request object.
Definition exception.h:105
static constexpr int PARSE_ERROR
Invalid JSON was received by the server.
Definition exception.h:100
static constexpr std::string_view err_method_not_found
Error message for when the method is not found.
Definition exception.h:50
static constexpr std::string_view err_not_jsonrpc_2_0_request
Error message for when the request is not a JSON-RPC 2.0 request.
Definition exception.h:31
static constexpr std::string_view err_empty_batch
Error message for when the batch request is empty.
Definition exception.h:77
static constexpr std::string_view err_empty_method
Error message for when the method is empty.
Definition exception.h:56
static constexpr std::string_view err_bad_request
Error request for when the request is not valid.
Definition exception.h:38
static constexpr std::string_view err_bad_params_type
Error message for when the parameters are not an array or an object.
Definition exception.h:62
static constexpr std::string_view err_invalid_params_passed_to_method
Error message for when the parameters passed to the method are not correct.
Definition exception.h:44
static constexpr std::string_view err_bad_id_type
Error message for when the ID is not a number, a string, or null.
Definition exception.h:70
constexpr auto convert_args(const nlohmann::json &params, std::index_sequence< Indices... >)
Converts JSON parameters to a tuple of arguments based on the specified types.
Definition details.h:335
Contains the implementation details of the JSON RPC library.
Definition details.h:31
Represents a JSON RPC request.
Definition request.h:22