1pub mod jsonrpc_types;
20
21mod parser;
22mod util;
23
24use crate::lotus_json::HasLotusJson;
25
26use self::{jsonrpc_types::RequestParameters, util::Optional as _};
27use super::error::ServerError as Error;
28use ahash::HashMap;
29use anyhow::Context as _;
30use enumflags2::{BitFlags, bitflags, make_bitflags};
31use fvm_ipld_blockstore::Blockstore;
32use http::{Extensions, Uri};
33use jsonrpsee::RpcModule;
34use openrpc_types::{ContentDescriptor, Method, ParamStructure, ReferenceOr};
35use parser::Parser;
36use schemars::{JsonSchema, Schema, SchemaGenerator};
37use serde::{
38 Deserialize, Serialize,
39 de::{Error as _, Unexpected},
40};
41use std::{future::Future, str::FromStr, sync::Arc};
42use strum::EnumString;
43
44pub type Ctx<T> = Arc<crate::rpc::RPCState<T>>;
46
47pub trait RpcMethod<const ARITY: usize> {
59 const N_REQUIRED_PARAMS: usize = ARITY;
61 const NAME: &'static str;
63 const NAME_ALIAS: Option<&'static str> = None;
65 const PARAM_NAMES: [&'static str; ARITY];
67 const API_PATHS: BitFlags<ApiPaths>;
69 const PERMISSION: Permission;
71 const SUMMARY: Option<&'static str> = None;
73 const DESCRIPTION: Option<&'static str> = None;
75 type Params: Params<ARITY>;
77 type Ok: HasLotusJson;
79 fn handle(
81 ctx: Ctx<impl Blockstore + Send + Sync + 'static>,
82 params: Self::Params,
83 ext: &Extensions,
84 ) -> impl Future<Output = Result<Self::Ok, Error>> + Send;
85 const SUBSCRIPTION: bool = false;
87}
88
89#[derive(
91 Debug,
92 Clone,
93 Copy,
94 PartialEq,
95 Eq,
96 PartialOrd,
97 Ord,
98 Hash,
99 derive_more::Display,
100 Serialize,
101 Deserialize,
102)]
103#[serde(rename_all = "snake_case")]
104pub enum Permission {
105 Admin,
107 Sign,
109 Write,
111 Read,
113}
114
115#[bitflags]
119#[repr(u8)]
120#[derive(
121 Debug,
122 Default,
123 Clone,
124 Copy,
125 Hash,
126 Eq,
127 PartialEq,
128 Ord,
129 PartialOrd,
130 clap::ValueEnum,
131 EnumString,
132 Deserialize,
133 Serialize,
134)]
135pub enum ApiPaths {
136 #[strum(ascii_case_insensitive)]
138 V0 = 0b00000001,
139 #[strum(ascii_case_insensitive)]
141 #[default]
142 V1 = 0b00000010,
143 #[strum(ascii_case_insensitive)]
145 V2 = 0b00000100,
146}
147
148impl ApiPaths {
149 pub const fn all() -> BitFlags<Self> {
150 make_bitflags!(Self::{ V0 | V1 })
152 }
153
154 pub const fn all_with_v2() -> BitFlags<Self> {
158 Self::all().union_c(make_bitflags!(Self::{ V2 }))
159 }
160
161 pub fn from_uri(uri: &Uri) -> anyhow::Result<Self> {
162 Ok(Self::from_str(uri.path().trim_start_matches("/rpc/"))?)
163 }
164
165 pub fn path(&self) -> &'static str {
166 match self {
167 Self::V0 => "rpc/v0",
168 Self::V1 => "rpc/v1",
169 Self::V2 => "rpc/v2",
170 }
171 }
172}
173
174pub trait RpcMethodExt<const ARITY: usize>: RpcMethod<ARITY> {
177 fn build_params(
181 params: Self::Params,
182 calling_convention: ConcreteCallingConvention,
183 ) -> Result<RequestParameters, serde_json::Error> {
184 let args = params.unparse()?;
185 match calling_convention {
186 ConcreteCallingConvention::ByPosition => {
187 Ok(RequestParameters::ByPosition(Vec::from(args)))
188 }
189 ConcreteCallingConvention::ByName => Ok(RequestParameters::ByName(
190 itertools::zip_eq(Self::PARAM_NAMES.into_iter().map(String::from), args).collect(),
191 )),
192 }
193 }
194
195 fn parse_params(
196 params_raw: Option<impl AsRef<str>>,
197 calling_convention: ParamStructure,
198 ) -> anyhow::Result<Self::Params> {
199 Ok(Self::Params::parse(
200 params_raw
201 .map(|s| serde_json::from_str(s.as_ref()))
202 .transpose()?,
203 Self::PARAM_NAMES,
204 calling_convention,
205 Self::N_REQUIRED_PARAMS,
206 )?)
207 }
208
209 fn openrpc<'de>(
211 g: &mut SchemaGenerator,
212 calling_convention: ParamStructure,
213 method_name: &'static str,
214 ) -> Method
215 where
216 <Self::Ok as HasLotusJson>::LotusJson: JsonSchema + Deserialize<'de>,
217 {
218 Method {
219 name: String::from(method_name),
220 params: itertools::zip_eq(Self::PARAM_NAMES, Self::Params::schemas(g))
221 .enumerate()
222 .map(|(pos, (name, (schema, nullable)))| {
223 let required = pos <= Self::N_REQUIRED_PARAMS;
224 if !required && !nullable {
225 panic!("Optional parameter at position {pos} should be of an optional type. method={method_name}, param_name={name}");
226 }
227 ReferenceOr::Item(ContentDescriptor {
228 name: String::from(name),
229 schema,
230 required: Some(required),
231 ..Default::default()
232 })
233 })
234 .collect(),
235 param_structure: Some(calling_convention),
236 result: Some(ReferenceOr::Item(ContentDescriptor {
237 name: format!("{method_name}.Result"),
238 schema: g.subschema_for::<<Self::Ok as HasLotusJson>::LotusJson>(),
239 required: Some(!<Self::Ok as HasLotusJson>::LotusJson::optional()),
240 ..Default::default()
241 })),
242 summary: Self::SUMMARY.map(Into::into),
243 description: Self::DESCRIPTION.map(Into::into),
244 ..Default::default()
245 }
246 }
247
248 fn register(
250 modules: &mut HashMap<
251 ApiPaths,
252 RpcModule<crate::rpc::RPCState<impl Blockstore + Send + Sync + 'static>>,
253 >,
254 calling_convention: ParamStructure,
255 ) -> Result<(), jsonrpsee::core::RegisterMethodError>
256 where
257 <Self::Ok as HasLotusJson>::LotusJson: Clone + 'static,
258 {
259 use clap::ValueEnum as _;
260
261 assert!(
262 Self::N_REQUIRED_PARAMS <= ARITY,
263 "N_REQUIRED_PARAMS({}) can not be greater than ARITY({ARITY}) in {}",
264 Self::N_REQUIRED_PARAMS,
265 Self::NAME
266 );
267
268 for api_version in ApiPaths::value_variants() {
269 if Self::API_PATHS.contains(*api_version)
270 && let Some(module) = modules.get_mut(api_version)
271 {
272 module.register_async_method(
273 Self::NAME,
274 move |params, ctx, extensions| async move {
275 let params = Self::parse_params(params.as_str(), calling_convention)
276 .map_err(|e| Error::invalid_params(e, None))?;
277 let ok = Self::handle(ctx, params, &extensions).await?;
278 Result::<_, jsonrpsee::types::ErrorObjectOwned>::Ok(ok.into_lotus_json())
279 },
280 )?;
281 if let Some(alias) = Self::NAME_ALIAS {
282 module.register_alias(alias, Self::NAME)?
283 }
284 }
285 }
286 Ok(())
287 }
288 fn request(params: Self::Params) -> Result<crate::rpc::Request<Self::Ok>, serde_json::Error> {
290 let params = Self::request_params(params)?;
292 Ok(crate::rpc::Request {
293 method_name: Self::NAME.into(),
294 params,
295 result_type: std::marker::PhantomData,
296 api_paths: Self::API_PATHS,
297 timeout: *crate::rpc::DEFAULT_REQUEST_TIMEOUT,
298 })
299 }
300
301 fn request_params(params: Self::Params) -> Result<serde_json::Value, serde_json::Error> {
302 Ok(
304 match Self::build_params(params, ConcreteCallingConvention::ByPosition)? {
305 RequestParameters::ByPosition(mut it) => {
306 while Self::N_REQUIRED_PARAMS < it.len() {
310 match it.last() {
311 Some(last) if last.is_null() => it.pop(),
312 _ => break,
313 };
314 }
315 serde_json::Value::Array(it)
316 }
317 RequestParameters::ByName(it) => serde_json::Value::Object(it),
318 },
319 )
320 }
321
322 fn request_with_alias(
324 params: Self::Params,
325 use_alias: bool,
326 ) -> anyhow::Result<crate::rpc::Request<Self::Ok>> {
327 let params = Self::request_params(params)?;
328 let name = if use_alias {
329 Self::NAME_ALIAS.context("alias is None")?
330 } else {
331 Self::NAME
332 };
333
334 Ok(crate::rpc::Request {
335 method_name: name.into(),
336 params,
337 result_type: std::marker::PhantomData,
338 api_paths: Self::API_PATHS,
339 timeout: *crate::rpc::DEFAULT_REQUEST_TIMEOUT,
340 })
341 }
342 fn call_raw(
343 client: &crate::rpc::client::Client,
344 params: Self::Params,
345 ) -> impl Future<Output = Result<<Self::Ok as HasLotusJson>::LotusJson, jsonrpsee::core::ClientError>>
346 {
347 async {
348 let json = client.call(Self::request(params)?.map_ty()).await?;
352 Ok(serde_json::from_value(json)?)
353 }
354 }
355 fn call(
356 client: &crate::rpc::client::Client,
357 params: Self::Params,
358 ) -> impl Future<Output = Result<Self::Ok, jsonrpsee::core::ClientError>> {
359 async {
360 Self::call_raw(client, params)
361 .await
362 .map(Self::Ok::from_lotus_json)
363 }
364 }
365}
366impl<const ARITY: usize, T> RpcMethodExt<ARITY> for T where T: RpcMethod<ARITY> {}
367
368pub trait Params<const ARITY: usize>: HasLotusJson {
372 fn schemas(g: &mut SchemaGenerator) -> [(Schema, bool); ARITY];
375 fn parse(
378 raw: Option<RequestParameters>,
379 names: [&str; ARITY],
380 calling_convention: ParamStructure,
381 n_required: usize,
382 ) -> Result<Self, Error>
383 where
384 Self: Sized;
385 fn unparse(self) -> Result<[serde_json::Value; ARITY], serde_json::Error> {
389 match self.into_lotus_json_value() {
390 Ok(serde_json::Value::Array(args)) => match args.try_into() {
391 Ok(it) => Ok(it),
392 Err(_) => Err(serde_json::Error::custom("ARITY mismatch")),
393 },
394 Ok(serde_json::Value::Null) if ARITY == 0 => {
395 Ok(std::array::from_fn(|_ix| Default::default()))
396 }
397 Ok(it) => Err(serde_json::Error::invalid_type(
398 unexpected(&it),
399 &"a Vec with an item for each argument",
400 )),
401 Err(e) => Err(e),
402 }
403 }
404}
405
406fn unexpected(v: &serde_json::Value) -> Unexpected<'_> {
407 match v {
408 serde_json::Value::Null => Unexpected::Unit,
409 serde_json::Value::Bool(it) => Unexpected::Bool(*it),
410 serde_json::Value::Number(it) => match (it.as_f64(), it.as_i64(), it.as_u64()) {
411 (None, None, None) => Unexpected::Other("Number"),
412 (Some(it), _, _) => Unexpected::Float(it),
413 (_, Some(it), _) => Unexpected::Signed(it),
414 (_, _, Some(it)) => Unexpected::Unsigned(it),
415 },
416 serde_json::Value::String(it) => Unexpected::Str(it),
417 serde_json::Value::Array(_) => Unexpected::Seq,
418 serde_json::Value::Object(_) => Unexpected::Map,
419 }
420}
421
422macro_rules! do_impls {
423 ($arity:literal $(, $arg:ident)* $(,)?) => {
424 const _: () = {
425 let _assert: [&str; $arity] = [$(stringify!($arg)),*];
426 };
427
428 impl<$($arg),*> Params<$arity> for ($($arg,)*)
429 where
430 $($arg: HasLotusJson + Clone, <$arg as HasLotusJson>::LotusJson: JsonSchema, )*
431 {
432 fn parse(
433 raw: Option<RequestParameters>,
434 arg_names: [&str; $arity],
435 calling_convention: ParamStructure,
436 n_required: usize,
437 ) -> Result<Self, Error> {
438 let mut _parser = Parser::new(raw, &arg_names, calling_convention, n_required)?;
439 Ok(($(_parser.parse::<crate::lotus_json::LotusJson<$arg>>()?.into_inner(),)*))
440 }
441 fn schemas(_gen: &mut SchemaGenerator) -> [(Schema, bool); $arity] {
442 [$((_gen.subschema_for::<$arg::LotusJson>(), $arg::LotusJson::optional())),*]
443 }
444 }
445 };
446}
447
448do_impls!(0);
449do_impls!(1, T0);
450do_impls!(2, T0, T1);
451do_impls!(3, T0, T1, T2);
452do_impls!(4, T0, T1, T2, T3);
453pub enum ConcreteCallingConvention {
463 ByPosition,
464 #[allow(unused)] ByName,
466}
467
468#[cfg(test)]
469mod tests {
470 use super::*;
471
472 #[test]
473 fn test_api_paths_from_uri() {
474 let v0 = ApiPaths::from_uri(&"http://127.0.0.1:2345/rpc/v0".parse().unwrap()).unwrap();
475 assert_eq!(v0, ApiPaths::V0);
476 let v1 = ApiPaths::from_uri(&"http://127.0.0.1:2345/rpc/v1".parse().unwrap()).unwrap();
477 assert_eq!(v1, ApiPaths::V1);
478 let v2 = ApiPaths::from_uri(&"http://127.0.0.1:2345/rpc/v2".parse().unwrap()).unwrap();
479 assert_eq!(v2, ApiPaths::V2);
480
481 ApiPaths::from_uri(&"http://127.0.0.1:2345/rpc/v3".parse().unwrap()).unwrap_err();
482 }
483}