-
Notifications
You must be signed in to change notification settings - Fork 176
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
rpc module: report error on invalid subscription #561
Changes from 3 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -44,6 +44,13 @@ pub struct RpcError<'a> { | |
pub id: Id<'a>, | ||
} | ||
|
||
impl<'a> RpcError<'a> { | ||
/// Create a new `RpcError`. | ||
pub fn new(error: ErrorObject<'a>, id: Id<'a>) -> Self { | ||
Self { jsonrpc: TwoPointZero, error, id } | ||
} | ||
} | ||
|
||
impl<'a> fmt::Display for RpcError<'a> { | ||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||
write!(f, "{}", serde_json::to_string(&self).expect("infallible; qed")) | ||
|
@@ -64,6 +71,13 @@ pub struct ErrorObject<'a> { | |
pub data: Option<&'a RawValue>, | ||
} | ||
|
||
impl<'a> ErrorObject<'a> { | ||
/// Create a new `ErrorObject` with optional data. | ||
pub fn new(code: ErrorCode, data: Option<&'a RawValue>) -> ErrorObject<'a> { | ||
Self { code, message: code.message(), data } | ||
} | ||
} | ||
|
||
impl<'a> From<ErrorCode> for ErrorObject<'a> { | ||
fn from(code: ErrorCode) -> Self { | ||
Self { code, message: code.message(), data: None } | ||
|
@@ -73,7 +87,7 @@ impl<'a> From<ErrorCode> for ErrorObject<'a> { | |
impl<'a> PartialEq for ErrorObject<'a> { | ||
fn eq(&self, other: &Self) -> bool { | ||
let this_raw = self.data.map(|r| r.get()); | ||
let other_raw = self.data.map(|r| r.get()); | ||
let other_raw = other.data.map(|r| r.get()); | ||
self.code == other.code && self.message == other.message && this_raw == other_raw | ||
} | ||
} | ||
|
@@ -84,6 +98,8 @@ pub const PARSE_ERROR_CODE: i32 = -32700; | |
pub const OVERSIZED_REQUEST_CODE: i32 = -32701; | ||
/// Oversized response error code. | ||
pub const OVERSIZED_RESPONSE_CODE: i32 = -32702; | ||
/// Oversized response error code. | ||
pub const INVALID_SUBSCRIPTION_CODE: i32 = -32703; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. BTW, technically these error codes should be used for application errors. To be really nitpicky I guess we should use There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Given this is a new code, do you think we should do the right thing for new error codes and stay within the right range? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. yeah, I used |
||
/// Internal error code. | ||
pub const INTERNAL_ERROR_CODE: i32 = -32603; | ||
/// Invalid params error code. | ||
|
@@ -105,6 +121,8 @@ pub const PARSE_ERROR_MSG: &str = "Parse error"; | |
pub const OVERSIZED_REQUEST_MSG: &str = "Request is too big"; | ||
/// Oversized response message | ||
pub const OVERSIZED_RESPONSE_MSG: &str = "Response is too big"; | ||
/// Invalid subscription ID. | ||
pub const INVALID_SUBSCRIPTION_MSG: &str = "Invalid subscription ID"; | ||
/// Internal error message. | ||
pub const INTERNAL_ERROR_MSG: &str = "Internal error"; | ||
/// Invalid params error message. | ||
|
@@ -212,6 +230,11 @@ impl serde::Serialize for ErrorCode { | |
} | ||
} | ||
|
||
/// Create a invalid subscription ID error. | ||
pub fn invalid_subscription_err(data: Option<&RawValue>) -> ErrorObject { | ||
ErrorObject::new(ErrorCode::ServerError(INVALID_SUBSCRIPTION_CODE), data) | ||
} | ||
|
||
#[cfg(test)] | ||
mod tests { | ||
use super::{ErrorCode, ErrorObject, Id, RpcError, TwoPointZero}; | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -27,12 +27,16 @@ | |
#![cfg(test)] | ||
|
||
use crate::types::error::{CallError, Error}; | ||
use crate::types::v2::{self, Response, RpcError}; | ||
use crate::types::DeserializeOwned; | ||
use crate::{future::ServerHandle, RpcModule, WsServerBuilder}; | ||
use anyhow::anyhow; | ||
use futures_util::future::join; | ||
use jsonrpsee_test_utils::helpers::*; | ||
use jsonrpsee_test_utils::mocks::{Id, TestContext, WebSocketTestClient, WebSocketTestError}; | ||
use jsonrpsee_test_utils::TimeoutFutureExt; | ||
use jsonrpsee_types::to_json_raw_value; | ||
use jsonrpsee_types::v2::error::invalid_subscription_err; | ||
use serde_json::Value as JsonValue; | ||
use std::{fmt, net::SocketAddr, time::Duration}; | ||
use tracing_subscriber::{EnvFilter, FmtSubscriber}; | ||
|
@@ -41,6 +45,11 @@ fn init_logger() { | |
let _ = FmtSubscriber::builder().with_env_filter(EnvFilter::from_default_env()).try_init(); | ||
} | ||
|
||
fn deser_call<T: DeserializeOwned>(raw: String) -> T { | ||
let out: Response<T> = serde_json::from_str(&raw).unwrap(); | ||
out.result | ||
} | ||
|
||
/// Applications can/should provide their own error. | ||
#[derive(Debug)] | ||
struct MyAppError; | ||
|
@@ -107,6 +116,15 @@ async fn server_with_handles() -> (SocketAddr, ServerHandle) { | |
Ok("Yawn!") | ||
}) | ||
.unwrap(); | ||
module | ||
.register_subscription("subscribe_hello", "unsubscribe_hello", |_, sink, _| { | ||
std::thread::spawn(move || loop { | ||
let _ = sink; | ||
std::thread::sleep(std::time::Duration::from_secs(30)); | ||
}); | ||
Ok(()) | ||
}) | ||
.unwrap(); | ||
|
||
let addr = server.local_addr().unwrap(); | ||
|
||
|
@@ -569,3 +587,38 @@ async fn run_forever() { | |
// Send the shutdown request from one handle and await the server on the second one. | ||
join(server_handle.clone().stop().unwrap(), server_handle).with_timeout(TIMEOUT).await.unwrap(); | ||
} | ||
|
||
#[tokio::test] | ||
async fn unsubscribe_twice_should_indicate_error() { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ideally these tests should be in the RPC module but I have to refactor the |
||
init_logger(); | ||
let addr = server().await; | ||
let mut client = WebSocketTestClient::new(addr).with_default_timeout().await.unwrap().unwrap(); | ||
|
||
let sub_call = call("subscribe_hello", Vec::<()>::new(), Id::Num(0)); | ||
let sub_id: u64 = deser_call(client.send_request_text(sub_call).await.unwrap()); | ||
|
||
let unsub_call = call("unsubscribe_hello", vec![sub_id], Id::Num(1)); | ||
let unsub_1: String = deser_call(client.send_request_text(unsub_call).await.unwrap()); | ||
assert_eq!(&unsub_1, "Unsubscribed"); | ||
|
||
let unsub_call = call("unsubscribe_hello", vec![sub_id], Id::Num(2)); | ||
let unsub_2 = client.send_request_text(unsub_call).await.unwrap(); | ||
let unsub_2_err: RpcError = serde_json::from_str(&unsub_2).unwrap(); | ||
let sub_id = to_json_raw_value(&sub_id).unwrap(); | ||
|
||
let err = Some(to_json_raw_value(&format!("Invalid subscription ID={}", sub_id)).unwrap()); | ||
assert_eq!(unsub_2_err, RpcError::new(invalid_subscription_err(err.as_deref()), v2::Id::Number(2))); | ||
} | ||
|
||
#[tokio::test] | ||
async fn unsubscribe_wrong_sub_id_type() { | ||
init_logger(); | ||
let addr = server().await; | ||
let mut client = WebSocketTestClient::new(addr).with_default_timeout().await.unwrap().unwrap(); | ||
|
||
let unsub = | ||
client.send_request_text(call("unsubscribe_hello", vec!["string_is_not_supported"], Id::Num(0))).await.unwrap(); | ||
let unsub_2_err: RpcError = serde_json::from_str(&unsub).unwrap(); | ||
let err = Some(to_json_raw_value(&"Invalid subscription ID type, must be integer").unwrap()); | ||
assert_eq!(unsub_2_err, RpcError::new(invalid_subscription_err(err.as_deref()), v2::Id::Number(0))); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
unrelated bug I found
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oops. No test for this eh?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Indeed