Skip to content
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

ws client redirections #397

Merged
merged 43 commits into from
Oct 5, 2021
Merged
Show file tree
Hide file tree
Changes from 40 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
31ae533
feat(ws client): support redirections
niklasad1 Jun 30, 2021
753699c
reuse socket
niklasad1 Jun 30, 2021
b6f09da
reuse socket
niklasad1 Jun 30, 2021
ce77bf5
add hacks
niklasad1 Jun 30, 2021
67b550e
fix build
niklasad1 Jun 30, 2021
e9d863f
remove hacks
niklasad1 Jun 30, 2021
4c1dc3c
Merge branch 'master' into na-ws-client-redirections
dvdplm Jul 12, 2021
8a31148
Merge remote-tracking branch 'origin/master' into na-ws-client-redire…
niklasad1 Aug 30, 2021
4685447
Merge branch 'na-ws-client-redirections' of github.com:paritytech/jso…
niklasad1 Aug 30, 2021
5a1e118
fix bad merge
niklasad1 Aug 30, 2021
9e506c1
address grumbles
niklasad1 Aug 30, 2021
ffa8150
Merge remote-tracking branch 'origin/master' into na-ws-client-redire…
niklasad1 Sep 29, 2021
d27a708
fix grumbles
niklasad1 Sep 29, 2021
1fce702
fix grumbles
niklasad1 Sep 29, 2021
250c896
fix nit
niklasad1 Sep 29, 2021
d200744
add redirection test
niklasad1 Sep 30, 2021
1f7cfe4
Merge branch 'master' into na-ws-client-redirections
dvdplm Oct 1, 2021
33f364c
Update test-utils/src/types.rs
dvdplm Oct 1, 2021
9d9aaba
Resolved todo
dvdplm Oct 1, 2021
720f709
Check that redirected client actually works
dvdplm Oct 1, 2021
ce87119
Merge branch 'master' into na-ws-client-redirections
dvdplm Oct 1, 2021
21583d1
Rename test-utils "types" to "mocks"
dvdplm Oct 1, 2021
f25a22d
Fix windows test (?)
dvdplm Oct 1, 2021
0d54d2e
fmt
dvdplm Oct 1, 2021
b93287c
What is wrong with you windows?
dvdplm Oct 1, 2021
4df383f
Ignore redirect test on windows
dvdplm Oct 1, 2021
ae2791c
fix bad transport errors
niklasad1 Oct 2, 2021
0d4b18b
debug windows tests
niklasad1 Oct 2, 2021
adbd2c6
update soketto
niklasad1 Oct 2, 2021
bf0d71d
maybe fix windows test
niklasad1 Oct 2, 2021
186bf84
add config flag for max redirections
niklasad1 Oct 2, 2021
efe1c28
revert faulty change.
niklasad1 Oct 2, 2021
e70842b
revert windows path
niklasad1 Oct 2, 2021
549e563
use manual join paths
niklasad1 Oct 3, 2021
df28dbd
remove url dep
niklasad1 Oct 3, 2021
4b5065c
Update ws-client/src/tests.rs
niklasad1 Oct 3, 2021
93d0c3b
default max redirects 5
niklasad1 Oct 4, 2021
c257fea
Merge branch 'na-ws-client-redirections' of github.com:paritytech/jso…
niklasad1 Oct 4, 2021
0488a2a
remove needless clone vec
niklasad1 Oct 5, 2021
56db40a
Merge remote-tracking branch 'origin/master' into na-ws-client-redire…
niklasad1 Oct 5, 2021
3aede64
Merge remote-tracking branch 'origin/master' into na-ws-client-redire…
niklasad1 Oct 5, 2021
223c37c
fix bad merge
niklasad1 Oct 5, 2021
ae1e405
cmon CI run
niklasad1 Oct 5, 2021
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion http-client/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ use crate::types::{
};
use crate::HttpClientBuilder;
use jsonrpsee_test_utils::helpers::*;
use jsonrpsee_test_utils::types::Id;
use jsonrpsee_test_utils::mocks::Id;
use jsonrpsee_test_utils::TimeoutFutureExt;

#[tokio::test]
Expand Down
2 changes: 1 addition & 1 deletion http-server/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ use crate::types::error::{CallError, Error};
use crate::{server::StopHandle, HttpServerBuilder, RpcModule};

use jsonrpsee_test_utils::helpers::*;
use jsonrpsee_test_utils::types::{Id, StatusCode, TestContext};
use jsonrpsee_test_utils::mocks::{Id, StatusCode, TestContext};
use jsonrpsee_test_utils::TimeoutFutureExt;
use serde_json::Value as JsonValue;
use tokio::task::JoinHandle;
Expand Down
2 changes: 1 addition & 1 deletion test-utils/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,6 @@ hyper = { version = "0.14.10", features = ["full"] }
log = "0.4"
serde = { version = "1", default-features = false, features = ["derive"] }
serde_json = "1"
soketto = "0.7"
soketto = { version = "0.7", features = ["http"] }
niklasad1 marked this conversation as resolved.
Show resolved Hide resolved
tokio = { version = "1", features = ["net", "rt-multi-thread", "macros", "time"] }
tokio-util = { version = "0.6", features = ["compat"] }
2 changes: 1 addition & 1 deletion test-utils/src/helpers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@
// IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.

use crate::types::{Body, HttpResponse, Id, Uri};
use crate::mocks::{Body, HttpResponse, Id, Uri};
use hyper::service::{make_service_fn, service_fn};
use hyper::{Request, Response, Server};
use serde_json::Value;
Expand Down
2 changes: 1 addition & 1 deletion test-utils/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ use std::{future::Future, time::Duration};
use tokio::time::{timeout, Timeout};

pub mod helpers;
pub mod types;
pub mod mocks;

/// Helper extension trait which allows to limit execution time for the futures.
/// It is helpful in tests to ensure that no future will ever get stuck forever.
Expand Down
63 changes: 59 additions & 4 deletions test-utils/src/types.rs → test-utils/src/mocks.rs
Original file line number Diff line number Diff line change
Expand Up @@ -34,10 +34,8 @@ use futures_util::{
stream::{self, StreamExt},
};
use serde::{Deserialize, Serialize};
use soketto::handshake::{self, server::Response, Error as SokettoError, Server};
use std::io;
use std::net::SocketAddr;
use std::time::Duration;
use soketto::handshake::{self, http::is_upgrade_request, server::Response, Error as SokettoError, Server};
use std::{io, net::SocketAddr, time::Duration};
use tokio::net::TcpStream;
use tokio_util::compat::{Compat, TokioAsyncReadCompatExt};

Expand Down Expand Up @@ -314,3 +312,60 @@ async fn connection_task(socket: tokio::net::TcpStream, mode: ServerMode, mut ex
}
}
}

// Run a WebSocket server running on localhost that redirects requests for testing.
// Requests to any url except for `/myblock/two` will redirect one or two times (HTTP 301) and eventually end up in `/myblock/two`.
pub fn ws_server_with_redirect(other_server: String) -> String {
let addr = ([127, 0, 0, 1], 0).into();

let service = hyper::service::make_service_fn(move |_| {
let other_server = other_server.clone();
async move {
Ok::<_, hyper::Error>(hyper::service::service_fn(move |req| {
let other_server = other_server.clone();
async move { handler(req, other_server).await }
}))
}
});
let server = hyper::Server::bind(&addr).serve(service);
let addr = server.local_addr();

tokio::spawn(async move { server.await });
format!("ws://{}", addr)
}

/// Handle incoming HTTP Requests.
async fn handler(
req: hyper::Request<Body>,
other_server: String,
) -> Result<hyper::Response<Body>, soketto::BoxedError> {
if is_upgrade_request(&req) {
log::debug!("{:?}", req);

match req.uri().path() {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ahh okay the above description makes sense after reading this! 👍

"/myblock/two" => {
let response = hyper::Response::builder()
.status(301)
.header("Location", other_server)
.body(Body::empty())
.unwrap();
Ok(response)
}
"/myblock/one" => {
let response =
hyper::Response::builder().status(301).header("Location", "two").body(Body::empty()).unwrap();
Ok(response)
}
_ => {
let response = hyper::Response::builder()
.status(301)
.header("Location", "/myblock/one")
.body(Body::empty())
.unwrap();
Ok(response)
}
}
} else {
panic!("expect upgrade to WS");
}
}
3 changes: 2 additions & 1 deletion ws-client/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -25,9 +25,10 @@ serde_json = "1"
soketto = "0.7"
pin-project = "1"
thiserror = "1"
url = "2"
http = "0.2"
Copy link
Member Author

@niklasad1 niklasad1 Oct 3, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Switched to the http crate instead for URI parsing because it supports both absolute URI and relative resource URI

Useful when getting the location header back...

However, it doesn't support to get the default port and it's put in an Option<Port> so you can't really tell whether the port was invalid or no port was provided....

rustls = "0.19.1"
rustls-native-certs = "0.5.0"

[dev-dependencies]
jsonrpsee-test-utils = { path = "../test-utils" }
env_logger = "0.9"
33 changes: 21 additions & 12 deletions ws-client/src/client.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@
// IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.

use crate::transport::{Receiver as WsReceiver, Sender as WsSender, Target, WsTransportClientBuilder};
use crate::transport::{Receiver as WsReceiver, Sender as WsSender, WsHandshakeError, WsTransportClientBuilder};
use crate::types::{
traits::{Client, SubscriptionClient},
v2::{Id, Notification, NotificationSer, ParamsSer, RequestSer, Response, RpcError, SubscriptionResponse},
Expand All @@ -46,10 +46,13 @@ use futures::{
prelude::*,
sink::SinkExt,
};
use http::uri::{InvalidUri, Uri};
use tokio::sync::Mutex;

use serde::de::DeserializeOwned;
use std::{borrow::Cow, time::Duration};
use std::{borrow::Cow, convert::TryInto, time::Duration};

pub use soketto::handshake::client::Header;

/// Wrapper over a [`oneshot::Receiver`](futures::channel::oneshot::Receiver) that reads
/// the underlying channel once and then stores the result in String.
Expand Down Expand Up @@ -109,6 +112,7 @@ pub struct WsClientBuilder<'a> {
origin_header: Option<Cow<'a, str>>,
max_concurrent_requests: usize,
max_notifs_per_subscription: usize,
max_redirections: usize,
}

impl<'a> Default for WsClientBuilder<'a> {
Expand All @@ -121,6 +125,7 @@ impl<'a> Default for WsClientBuilder<'a> {
origin_header: None,
max_concurrent_requests: 256,
max_notifs_per_subscription: 1024,
max_redirections: 5,
}
}
}
Expand Down Expand Up @@ -151,8 +156,8 @@ impl<'a> WsClientBuilder<'a> {
}

/// Set origin header to pass during the handshake.
pub fn origin_header(mut self, origin: &'a str) -> Self {
self.origin_header = Some(Cow::Borrowed(origin));
pub fn origin_header(mut self, origin: Cow<'a, str>) -> Self {
self.origin_header = Some(origin);
self
}

Expand All @@ -176,31 +181,35 @@ impl<'a> WsClientBuilder<'a> {
self
}

/// Set the max number of redirections to perform until a connection is regarded as failed.
pub fn max_redirections(mut self, redirect: usize) -> Self {
self.max_redirections = redirect;
self
}

/// Build the client with specified URL to connect to.
/// If the port number is missing from the URL, the default port number is used.
///
///
/// `ws://host` - port 80 is used
///
/// `wss://host` - port 443 is used
/// You must provide the port number in the URL.
///
/// ## Panics
///
/// Panics if being called outside of `tokio` runtime context.
pub async fn build(self, url: &'a str) -> Result<WsClient, Error> {
pub async fn build(self, uri: &'a str) -> Result<WsClient, Error> {
let certificate_store = self.certificate_store;
let max_capacity_per_subscription = self.max_notifs_per_subscription;
let max_concurrent_requests = self.max_concurrent_requests;
let request_timeout = self.request_timeout;
let (to_back, from_front) = mpsc::channel(self.max_concurrent_requests);
let (err_tx, err_rx) = oneshot::channel();

let uri: Uri = uri.parse().map_err(|e: InvalidUri| Error::Transport(e.into()))?;

let builder = WsTransportClientBuilder {
certificate_store,
target: Target::parse(url).map_err(|e| Error::Transport(e.into()))?,
target: uri.try_into().map_err(|e: WsHandshakeError| Error::Transport(e.into()))?,
timeout: self.connection_timeout,
origin_header: self.origin_header,
max_request_body_size: self.max_request_body_size,
max_redirections: self.max_redirections,
};

let (sender, receiver) = builder.build().await.map_err(|e| Error::Transport(e.into()))?;
Expand Down
33 changes: 32 additions & 1 deletion ws-client/src/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ use crate::types::{
};
use crate::WsClientBuilder;
use jsonrpsee_test_utils::helpers::*;
use jsonrpsee_test_utils::types::{Id, WebSocketTestServer};
use jsonrpsee_test_utils::mocks::{Id, WebSocketTestServer};
use jsonrpsee_test_utils::TimeoutFutureExt;
use serde_json::Value as JsonValue;

Expand Down Expand Up @@ -263,3 +263,34 @@ fn assert_error_response(err: Error, exp: ErrorObject) {
e => panic!("Expected error: \"{}\", got: {:?}", err, e),
};
}

#[tokio::test]
async fn redirections() {
let _ = env_logger::try_init();
let expected = "abc 123";
let server = WebSocketTestServer::with_hardcoded_response(
"127.0.0.1:0".parse().unwrap(),
ok_response(expected.into(), Id::Num(0)),
)
.with_default_timeout()
.await
.unwrap();

let server_url = format!("ws://{}", server.local_addr());
let redirect_url = jsonrpsee_test_utils::mocks::ws_server_with_redirect(server_url);

// The client will first connect to a server that only performs re-directions and finally
// redirect to another server to complete the handshake.
let client = WsClientBuilder::default().build(&redirect_url).with_default_timeout().await;
// It's an ok client
let client = match client {
Ok(Ok(client)) => client,
Ok(Err(e)) => panic!("WsClient builder failed with: {:?}", e),
Err(e) => panic!("WsClient builder timed out with: {:?}", e),
};
// It's connected
assert!(client.is_connected());
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

👍

// It works
let response = client.request::<String>("anything", ParamsSer::NoParams).with_default_timeout().await.unwrap();
assert_eq!(response.unwrap(), String::from(expected));
}
Loading