From dba977dfa2f93b1c0812122e12860ff4f62ad836 Mon Sep 17 00:00:00 2001 From: Erik De Smedt Date: Mon, 18 Dec 2023 21:42:11 +0100 Subject: [PATCH] Some basic documentation for `cln_rpc` --- cln-rpc/src/lib.rs | 92 +++++++++++++++++++++++++++++-- cln-rpc/src/model.rs | 19 ++++--- cln-rpc/src/primitives.rs | 1 + contrib/msggen/msggen/gen/rust.py | 19 ++++--- 4 files changed, 109 insertions(+), 22 deletions(-) diff --git a/cln-rpc/src/lib.rs b/cln-rpc/src/lib.rs index b83d28cabfec..99dbf5cb757e 100644 --- a/cln-rpc/src/lib.rs +++ b/cln-rpc/src/lib.rs @@ -1,6 +1,83 @@ +//! # A Core Lightning RPC-client +//! +//! Core Lightning exposes a JSON-RPC interface over unix-domain sockets. +//! The unix-domain socket appears like file and located by default in +//! `~/.lightning//lightning-rpc`. +//! +//! This crate contains an RPC-client called [ClnRpc] and models +//! for most [requests](crate::model::requests) and [responses](crate::model::responses). +//! +//! The example below shows how to initiate the client and celss the `getinfo`-rpc method. +//! +//! ```no_run +//! use std::path::Path; +//! use tokio_test; +//! use cln_rpc::{ClnRpc, TypedRequest}; +//! use cln_rpc::model::requests::GetinfoRequest; +//! use cln_rpc::model::responses::GetinfoResponse; +//! +//! tokio_test::block_on( async { +//! let path = Path::new("path_to_lightning_dir"); +//! let mut rpc = ClnRpc::new(path).await.unwrap(); +//! let request = GetinfoRequest {}; +//! let response : GetinfoResponse = rpc.call_typed(request).await.unwrap(); +//! }); +//! ``` +//! +//! If the required model is not available you can implement [`TypedRequest`] +//! and use [`ClnRpc::call_typed`] without a problem. +//! +//! ```no_run +//! use std::path::Path; +//! use tokio_test; +//! use cln_rpc::{ClnRpc, TypedRequest}; +//! use serde::{Serialize, Deserialize}; +//! +//! #[derive(Serialize, Debug)] +//! struct CustomMethodRequest { +//! param_a : String +//! }; +//! #[derive(Deserialize, Debug)] +//! struct CustomMethodResponse { +//! field_a : String +//! }; +//! +//! impl TypedRequest for CustomMethodRequest { +//! type Response = CustomMethodResponse; +//! +//! fn method(&self) -> &str { +//! "custommethod" +//! } +//! } +//! +//! tokio_test::block_on( async { +//! let path = Path::new("path_to_lightning_dir"); +//! let mut rpc = ClnRpc::new(path).await.unwrap(); +//! +//! let request = CustomMethodRequest { param_a : String::from("example")}; +//! let response = rpc.call_typed(request).await.unwrap(); +//! }) +//! ``` +//! +//! An alternative is to use [`ClnRpc::call_raw`]. +//! +//! ```no_run +//! use std::path::Path; +//! use tokio_test; +//! use cln_rpc::{ClnRpc, TypedRequest}; +//! +//! tokio_test::block_on( async { +//! let path = Path::new("path_to_lightning_dir"); +//! let mut rpc = ClnRpc::new(path).await.unwrap(); +//! let method = "custommethod"; +//! let request = serde_json::json!({"param_a" : "example"}); +//! let response : serde_json::Value = rpc.call_raw(method, request).await.unwrap(); +//! }) +//! ``` +//! use crate::codec::JsonCodec; pub use anyhow::Error; -use anyhow::Result; +use anyhow::Result; use core::fmt::Debug; use futures_util::sink::SinkExt; use futures_util::StreamExt; @@ -19,13 +96,16 @@ pub mod model; pub mod notifications; pub mod primitives; -use crate::model::TypedRequest; +pub use crate::model::TypedRequest; pub use crate::{ model::{Request, Response}, notifications::Notification, primitives::RpcError, }; +/// An RPC-client for Core Lightning +/// +/// /// pub struct ClnRpc { next_id: AtomicUsize, @@ -59,11 +139,11 @@ impl ClnRpc { /// Low-level API to call the rpc. /// - /// It is the responsibility of the caller to pick valid types `R` and `P`. - /// It's useful for ad-hoc calls to methods that are not present in [`crate::model`]. - /// Users can use [`serde_json::Value`] and don't have to implement any custom structs. + /// An interesting choice of `R` and `P` is [`serde_json::Value`] because it allows + /// ad-hoc calls to custom RPC-methods /// - /// Most users would prefer to use [call_typed](crate::ClnRpc::call_typed) instead. + /// If you are using a model such as [`crate::model::requests::GetinfoRequest`] you'd + /// probably want to use [`Self::call_typed`] instead. /// /// Example: /// ```no_run diff --git a/cln-rpc/src/model.rs b/cln-rpc/src/model.rs index a384a2538fee..48494be9f45c 100644 --- a/cln-rpc/src/model.rs +++ b/cln-rpc/src/model.rs @@ -1,13 +1,16 @@ #![allow(non_camel_case_types)] -//! This file was automatically generated using the following command: +// +// This file was automatically generated using the following command: +// +// ```bash +// contrib/msggen/msggen/__main__.py +// ``` +// +// Do not edit this file, it'll be overwritten. Rather edit the schema that +// this file was generated from + +//! A collection of models to describe [requests] and [responses]. //! -//! ```bash -//! contrib/msggen/msggen/__main__.py -//! ``` -//! -//! Do not edit this file, it'll be overwritten. Rather edit the schema that -//! this file was generated from - use serde::{Deserialize, Serialize}; #[derive(Clone, Debug, Serialize, Deserialize)] diff --git a/cln-rpc/src/primitives.rs b/cln-rpc/src/primitives.rs index 6a360a3718ce..659f623f5171 100644 --- a/cln-rpc/src/primitives.rs +++ b/cln-rpc/src/primitives.rs @@ -1,3 +1,4 @@ +//! Primitive types representing [`Amount`]s, [`PublicKey`]s, ... use anyhow::Context; use anyhow::{anyhow, Error, Result}; use bitcoin::hashes::Hash as BitcoinHash; diff --git a/contrib/msggen/msggen/gen/rust.py b/contrib/msggen/msggen/gen/rust.py index dc820ece4a48..a304830c7ec9 100644 --- a/contrib/msggen/msggen/gen/rust.py +++ b/contrib/msggen/msggen/gen/rust.py @@ -39,15 +39,18 @@ } header = f"""#![allow(non_camel_case_types)] -//! This file was automatically generated using the following command: +// +// This file was automatically generated using the following command: +// +// ```bash +// {" ".join(sys.argv)} +// ``` +// +// Do not edit this file, it'll be overwritten. Rather edit the schema that +// this file was generated from + +//! A collection of models to describe [requests] and [responses]. //! -//! ```bash -//! {" ".join(sys.argv)} -//! ``` -//! -//! Do not edit this file, it'll be overwritten. Rather edit the schema that -//! this file was generated from - """