1mod invalid;
7use serde::{Deserialize, Serialize};
8use tor_rpcbase as rpc;
9
10#[derive(Debug, Eq, PartialEq, Hash, Clone, Serialize, Deserialize)]
15#[serde(untagged)]
16pub(crate) enum RequestId {
17 Str(Box<str>),
22 Int(i64),
30}
31
32#[derive(Debug, Clone, Serialize, Deserialize, Default)]
37pub(crate) struct ReqMeta {
38 #[serde(default)]
41 pub(crate) updates: bool,
42
43 #[serde(default)]
46 pub(crate) require: Vec<String>,
47}
48
49#[derive(Debug, Deserialize)]
51pub(crate) struct Request {
52 pub(crate) id: RequestId,
56 pub(crate) obj: rpc::ObjectId,
58 #[serde(default)]
60 pub(crate) meta: ReqMeta,
61 #[serde(flatten)]
65 pub(crate) method: Box<dyn rpc::DeserMethod>,
66}
67
68#[derive(Debug, serde::Deserialize)]
72#[serde(untagged)]
73pub(crate) enum FlexibleRequest {
74 Valid(Request),
76 Invalid(invalid::InvalidRequest),
78 }
88
89#[derive(Debug, Serialize)]
91pub(crate) struct BoxedResponse {
92 #[serde(skip_serializing_if = "Option::is_none")]
99 pub(crate) id: Option<RequestId>,
100 #[serde(flatten)]
102 pub(crate) body: ResponseBody,
103}
104
105impl BoxedResponse {
106 pub(crate) fn from_error<E>(id: Option<RequestId>, error: E) -> Self
109 where
110 E: Into<rpc::RpcError>,
111 {
112 let error: rpc::RpcError = error.into();
113 let body = ResponseBody::Error(Box::new(error));
114 Self { id, body }
115 }
116}
117
118#[derive(Serialize)]
120pub(crate) enum ResponseBody {
121 #[serde(rename = "error")]
123 Error(Box<rpc::RpcError>),
124 #[serde(rename = "result")]
130 Success(Box<dyn erased_serde::Serialize + Send>),
131 #[serde(rename = "update")]
134 Update(Box<dyn erased_serde::Serialize + Send>),
135}
136
137impl ResponseBody {
138 #[cfg(test)]
141 pub(crate) fn is_final(&self) -> bool {
142 match self {
143 ResponseBody::Error(_) | ResponseBody::Success(_) => true,
144 ResponseBody::Update(_) => false,
145 }
146 }
147}
148
149impl From<rpc::RpcError> for ResponseBody {
150 fn from(inp: rpc::RpcError) -> ResponseBody {
151 ResponseBody::Error(Box::new(inp))
152 }
153}
154
155impl std::fmt::Debug for ResponseBody {
156 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
157 let json = |x| match serde_json::to_string(x) {
159 Ok(s) => s,
160 Err(e) => format!("«could not serialize: {}»", e),
161 };
162 match self {
163 Self::Error(arg0) => f.debug_tuple("Error").field(arg0).finish(),
164 Self::Update(arg0) => f.debug_tuple("Update").field(&json(arg0)).finish(),
165 Self::Success(arg0) => f.debug_tuple("Success").field(&json(arg0)).finish(),
166 }
167 }
168}
169
170#[cfg(test)]
171mod test {
172 #![allow(clippy::bool_assert_comparison)]
174 #![allow(clippy::clone_on_copy)]
175 #![allow(clippy::dbg_macro)]
176 #![allow(clippy::mixed_attributes_style)]
177 #![allow(clippy::print_stderr)]
178 #![allow(clippy::print_stdout)]
179 #![allow(clippy::single_char_pattern)]
180 #![allow(clippy::unwrap_used)]
181 #![allow(clippy::unchecked_time_subtraction)]
182 #![allow(clippy::useless_vec)]
183 #![allow(clippy::needless_pass_by_value)]
184 use super::*;
186 use derive_deftly::Deftly;
187 use tor_rpcbase::templates::*;
188
189 macro_rules! assert_dbg_eq {
194 ($a:expr, $b:expr) => {
195 assert_eq!(format!("{:?}", $a), format!("{:?}", $b));
196 };
197 }
198
199 #[derive(Debug, serde::Deserialize, Deftly)]
203 #[derive_deftly(DynMethod)]
204 #[deftly(rpc(method_name = "x-test:dummy"))]
205 struct DummyMethod {
206 #[serde(default)]
207 #[allow(dead_code)]
208 stuff: u64,
209 }
210
211 impl rpc::RpcMethod for DummyMethod {
212 type Output = DummyResponse;
213 type Update = rpc::NoUpdates;
214 }
215
216 #[derive(Serialize)]
217 struct DummyResponse {
218 hello: i64,
219 world: String,
220 }
221
222 #[test]
223 fn valid_requests() {
224 let parse_request = |s| match serde_json::from_str::<FlexibleRequest>(s) {
225 Ok(FlexibleRequest::Valid(req)) => req,
226 other => panic!("{:?}", other),
227 };
228
229 let r =
230 parse_request(r#"{"id": 7, "obj": "hello", "method": "x-test:dummy", "params": {} }"#);
231 assert_dbg_eq!(
232 r,
233 Request {
234 id: RequestId::Int(7),
235 obj: rpc::ObjectId::from("hello"),
236 meta: ReqMeta::default(),
237 method: Box::new(DummyMethod { stuff: 0 })
238 }
239 );
240 }
241
242 #[test]
243 fn invalid_requests() {
244 use crate::err::RequestParseError as RPE;
245 fn parsing_error(s: &str) -> RPE {
246 match serde_json::from_str::<FlexibleRequest>(s) {
247 Ok(FlexibleRequest::Invalid(req)) => req.error(),
248 x => panic!("Didn't expect {:?}", x),
249 }
250 }
251
252 macro_rules! expect_err {
253 ($p:pat, $e:expr) => {
254 let err = parsing_error($e);
255 assert!(matches!(err, $p), "Unexpected error type {:?}", err);
256 };
257 }
258
259 expect_err!(
260 RPE::IdMissing,
261 r#"{ "obj": "hello", "method": "x-test:dummy", "params": {} }"#
262 );
263 expect_err!(
264 RPE::IdType,
265 r#"{ "id": {}, "obj": "hello", "method": "x-test:dummy", "params": {} }"#
266 );
267 expect_err!(
268 RPE::ObjMissing,
269 r#"{ "id": 3, "method": "x-test:dummy", "params": {} }"#
270 );
271 expect_err!(
272 RPE::ObjType,
273 r#"{ "id": 3, "obj": 9, "method": "x-test:dummy", "params": {} }"#
274 );
275 expect_err!(
276 RPE::MethodMissing,
277 r#"{ "id": 3, "obj": "hello", "params": {} }"#
278 );
279 expect_err!(
280 RPE::MethodType,
281 r#"{ "id": 3, "obj": "hello", "method": [], "params": {} }"#
282 );
283 expect_err!(
284 RPE::MetaType,
285 r#"{ "id": 3, "obj": "hello", "meta": 7, "method": "x-test:dummy", "params": {} }"#
286 );
287 expect_err!(
288 RPE::MetaType,
289 r#"{ "id": 3, "obj": "hello", "meta": { "updates": 3}, "method": "x-test:dummy", "params": {} }"#
290 );
291 expect_err!(
292 RPE::NoSuchMethod,
293 r#"{ "id": 3, "obj": "hello", "method": "arti:this-is-not-a-method", "params": {} }"#
294 );
295 expect_err!(
296 RPE::MissingParams,
297 r#"{ "id": 3, "obj": "hello", "method": "x-test:dummy" }"#
298 );
299 expect_err!(
300 RPE::ParamType,
301 r#"{ "id": 3, "obj": "hello", "method": "x-test:dummy", "params": 7 }"#
302 );
303 }
304
305 #[test]
306 fn fmt_replies() {
307 let resp = BoxedResponse {
308 id: Some(RequestId::Int(7)),
309 body: ResponseBody::Success(Box::new(DummyResponse {
310 hello: 99,
311 world: "foo".into(),
312 })),
313 };
314 let s = serde_json::to_string(&resp).unwrap();
315 assert_eq!(s, r#"{"id":7,"result":{"hello":99,"world":"foo"}}"#);
319
320 let resp = BoxedResponse::from_error(
321 None,
322 rpc::RpcError::from(crate::err::RequestParseError::IdMissing),
323 );
324 let s = serde_json::to_string(&resp).unwrap();
325 assert_eq!(
327 s,
328 r#"{"error":{"message":"Request did not have any `id` field.","code":-32600,"kinds":["rpc:InvalidRequest"]}}"#
329 );
330 }
331
332 #[test]
333 fn response_body_is_final() {
334 let response_body_error = ResponseBody::from(rpc::RpcError::new(
335 "This is a test!".to_string(),
336 rpc::RpcErrorKind::ObjectNotFound,
337 ));
338 assert!(response_body_error.is_final());
339
340 let response_body_success = ResponseBody::Success(Box::new(DummyResponse {
341 hello: 99,
342 world: "foo".into(),
343 }));
344 assert!(response_body_success.is_final());
345
346 let response_body_update = ResponseBody::Update(Box::new(DummyResponse {
347 hello: 99,
348 world: "foo".into(),
349 }));
350 assert!(!response_body_update.is_final());
351 }
352}