Add client websockets helper

This commit is contained in:
Nikolay Kim 2020-05-03 20:20:59 +06:00
parent b142aed394
commit fff60731e1
5 changed files with 61 additions and 15 deletions

View file

@ -1,6 +1,6 @@
# Changes
## [0.1.15] - 2020-04-30
## [0.1.15] - 2020-05-03
* ntex::util: Refactor stream dispatcher
@ -8,6 +8,8 @@
* ntex::http: Fix upgrade service readiness check
* ntex::http: Add client websockets helper
* ntex::ws: Add stream and sink wrappers for ws protocol
* ntex::web: Add websockets helper

View file

@ -33,7 +33,7 @@ mod request;
mod response;
mod sender;
mod test;
mod ws;
pub mod ws;
pub use self::builder::ClientBuilder;
pub use self::connect::BoxedSocket;
@ -44,7 +44,6 @@ pub use self::request::ClientRequest;
pub use self::response::{ClientResponse, JsonBody, MessageBody};
pub use self::sender::SendClientRequest;
pub use self::test::TestResponse;
pub use self::ws::WebsocketsRequest;
use crate::http::error::HttpError;
use crate::http::{HeaderMap, Method, RequestHead, Uri};

View file

@ -6,15 +6,20 @@ use std::{fmt, str};
#[cfg(feature = "cookie")]
use coo_kie::{Cookie, CookieJar};
use futures::Stream;
use crate::codec::Framed;
use crate::codec::{AsyncRead, AsyncWrite, Framed};
use crate::http::error::HttpError;
use crate::http::header::{self, HeaderName, HeaderValue, AUTHORIZATION};
use crate::http::{ConnectionType, Method, StatusCode, Uri, Version};
use crate::http::{Payload, RequestHead};
use crate::rt::time::timeout;
use crate::service::{IntoService, Service};
use crate::util::framed::{Dispatcher, DispatcherError};
use crate::ws;
pub use crate::ws::{CloseCode, CloseReason, Frame, Message};
use super::connect::BoxedSocket;
use super::error::{InvalidUrl, SendRequestError, WsClientError};
use super::response::ClientResponse;
@ -402,6 +407,23 @@ impl fmt::Debug for WebsocketsRequest {
}
}
/// Start client websockets service.
pub async fn start<Io, T, F, Rx>(
framed: Framed<Io, ws::Codec>,
rx: Rx,
service: F,
) -> Result<(), DispatcherError<T::Error, ws::Codec>>
where
Io: AsyncRead + AsyncWrite + Unpin + 'static,
T: Service<Request = ws::Frame, Response = Option<ws::Message>>,
T::Error: 'static,
T::Future: 'static,
F: IntoService<T>,
Rx: Stream<Item = ws::Message> + Unpin + 'static,
{
Dispatcher::with(framed, Some(rx), service.into_service()).await
}
#[cfg(test)]
mod tests {
use super::*;

View file

@ -1,7 +1,7 @@
use std::error::Error as StdError;
use bytes::Bytes;
use futures::{Stream, TryStreamExt};
use futures::{Sink, Stream, TryStreamExt};
pub use crate::ws::{CloseCode, CloseReason, Frame, Message};
@ -17,9 +17,9 @@ pub type WebSocketsSink =
/// Do websocket handshake and start websockets service.
pub async fn start<T, F, S, Err>(
factory: F,
req: HttpRequest,
payload: S,
factory: F,
) -> Result<HttpResponse, Err>
where
T: ServiceFactory<
@ -34,18 +34,41 @@ where
S: Stream<Item = Result<Bytes, PayloadError>> + Unpin + 'static,
Err: From<T::InitError>,
Err: From<HandshakeError>,
{
let (tx, rx) = mpsc::channel();
start_with(req, payload, tx, rx, factory).await
}
/// Do websocket handshake and start websockets service.
pub async fn start_with<T, F, S, Err, Tx, Rx>(
req: HttpRequest,
payload: S,
tx: Tx,
rx: Rx,
factory: F,
) -> Result<HttpResponse, Err>
where
T: ServiceFactory<
Config = ws::StreamEncoder<Tx>,
Request = Frame,
Response = Option<Message>,
>,
T::Error: StdError + 'static,
T::InitError: 'static,
T::Service: 'static,
F: IntoServiceFactory<T>,
S: Stream<Item = Result<Bytes, PayloadError>> + Unpin + 'static,
Err: From<T::InitError>,
Err: From<HandshakeError>,
Tx: Sink<Result<Bytes, Box<dyn StdError>>> + Clone + Unpin + 'static,
Tx::Error: StdError,
Rx: Stream<Item = Result<Bytes, Box<dyn StdError>>> + Unpin + 'static,
{
// ws handshake
let mut res = handshake(req.head())?;
let payload = payload.map_err(|e| {
let e: Box<dyn StdError> = Box::new(e);
e
});
// response body stream
let (tx, rx): (_, mpsc::Receiver<Result<Bytes, Box<dyn StdError>>>) =
mpsc::channel();
let sink = ws::StreamEncoder::new(tx);
// create ws service

View file

@ -25,11 +25,11 @@ async fn web_ws() {
App::new().service(web::resource("/").route(web::to(
|req: HttpRequest, pl: web::types::Payload| async move {
ws::start::<_, _, _, web::Error>(
req,
pl,
fn_factory_with_config(|_| async {
Ok::<_, web::Error>(fn_service(service))
}),
req,
pl,
)
.await
},