Move the websocket code to a separate ws feature (#320)

This commit is contained in:
Sergey Vilgelm 2024-03-28 02:00:43 -07:00 committed by GitHub
parent 4ed6712ac4
commit d3499feb80
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
9 changed files with 43 additions and 17 deletions

View file

@ -31,10 +31,10 @@ jobs:
run: cargo llvm-cov clean --workspace run: cargo llvm-cov clean --workspace
- name: Code coverage (glommio) - name: Code coverage (glommio)
run: cargo +nightly llvm-cov --no-report --all --no-default-features --features="glommio,cookie,url,compress,openssl,rustls" run: cargo +nightly llvm-cov --no-report --all --no-default-features --features="glommio,cookie,url,compress,openssl,rustls,ws"
- name: Code coverage - name: Code coverage
run: cargo +nightly llvm-cov --no-report --all --doctests --no-default-features --features="tokio,cookie,url,compress,openssl,rustls" run: cargo +nightly llvm-cov --no-report --all --doctests --no-default-features --features="tokio,cookie,url,compress,openssl,rustls,ws"
- name: Generate coverage report - name: Generate coverage report
run: cargo +nightly llvm-cov report --lcov --output-path lcov.info --ignore-filename-regex="ntex-tokio|ntex-glommio|ntex-async-std" run: cargo +nightly llvm-cov report --lcov --output-path lcov.info --ignore-filename-regex="ntex-tokio|ntex-glommio|ntex-async-std"

View file

@ -59,7 +59,7 @@ jobs:
continue-on-error: true continue-on-error: true
run: | run: |
cd ntex cd ntex
cargo test --no-default-features --no-fail-fast --features="async-std,cookie,url,compress,openssl,rustls" cargo test --no-default-features --no-fail-fast --features="async-std,cookie,url,compress,openssl,rustls,ws"
- name: Install cargo-cache - name: Install cargo-cache
continue-on-error: true continue-on-error: true

View file

@ -7,9 +7,12 @@ readme = "README.md"
keywords = ["ntex", "networking", "framework", "async", "futures"] keywords = ["ntex", "networking", "framework", "async", "futures"]
repository = "https://github.com/ntex-rs/ntex.git" repository = "https://github.com/ntex-rs/ntex.git"
documentation = "https://docs.rs/ntex/" documentation = "https://docs.rs/ntex/"
categories = ["network-programming", "asynchronous", categories = [
"web-programming::http-server", "network-programming",
"web-programming::websocket"] "asynchronous",
"web-programming::http-server",
"web-programming::websocket",
]
license = "MIT OR Apache-2.0" license = "MIT OR Apache-2.0"
edition = "2021" edition = "2021"
@ -47,6 +50,9 @@ glommio = ["ntex-net/glommio"]
# async-std runtime # async-std runtime
async-std = ["ntex-net/async-std"] async-std = ["ntex-net/async-std"]
# websocket support
ws = ["dep:sha-1"]
[dependencies] [dependencies]
ntex-codec = "0.6.2" ntex-codec = "0.6.2"
ntex-http = "0.1.12" ntex-http = "0.1.12"
@ -65,11 +71,14 @@ ntex-tls = "1.0.0"
base64 = "0.22" base64 = "0.22"
bitflags = "2" bitflags = "2"
log = "0.4" log = "0.4"
nanorand = { version = "0.7", default-features = false, features = ["std", "wyrand"] } nanorand = { version = "0.7", default-features = false, features = [
"std",
"wyrand",
] }
pin-project-lite = "0.2" pin-project-lite = "0.2"
regex = { version = "1.10", default-features = false, features = ["std"] } regex = { version = "1.10", default-features = false, features = ["std"] }
serde = { version = "1.0", features=["derive"] } serde = { version = "1.0", features = ["derive"] }
sha-1 = "0.10" sha-1 = { version = "0.10", optional = true }
thiserror = "1.0" thiserror = "1.0"
# http/web framework # http/web framework
@ -84,14 +93,14 @@ url-pkg = { version = "2.4", package = "url", optional = true }
coo-kie = { version = "0.18", package = "cookie", optional = true } coo-kie = { version = "0.18", package = "cookie", optional = true }
# openssl # openssl
tls-openssl = { version="0.10", package = "openssl", optional = true } tls-openssl = { version = "0.10", package = "openssl", optional = true }
# rustls # rustls
tls-rustls = { version = "0.23", package = "rustls", optional = true } tls-rustls = { version = "0.23", package = "rustls", optional = true }
webpki-roots = { version = "0.26", optional = true } webpki-roots = { version = "0.26", optional = true }
# compression # compression
brotli2 = { version="0.3.2", optional = true } brotli2 = { version = "0.3.2", optional = true }
flate2 = { version = "1.0.22", optional = true } flate2 = { version = "1.0.22", optional = true }
[dev-dependencies] [dev-dependencies]
@ -99,7 +108,7 @@ env_logger = "0.11"
rand = "0.8" rand = "0.8"
time = "0.3" time = "0.3"
futures-util = "0.3" futures-util = "0.3"
tls-openssl = { version="0.10", package = "openssl" } tls-openssl = { version = "0.10", package = "openssl" }
tls-rustls = { version = "0.23", package="rustls" } tls-rustls = { version = "0.23", package = "rustls" }
rustls-pemfile = "2" rustls-pemfile = "2"
webpki-roots = "0.26" webpki-roots = "0.26"

View file

@ -71,6 +71,7 @@ impl ClientResponse {
} }
} }
#[cfg(feature = "ws")]
pub(crate) fn with_empty_payload(head: ResponseHead, config: Rc<ClientConfig>) -> Self { pub(crate) fn with_empty_payload(head: ResponseHead, config: Rc<ClientConfig>) -> Self {
ClientResponse::new(head, Payload::None, config) ClientResponse::new(head, Payload::None, config)
} }

View file

@ -4,7 +4,10 @@ use std::{net, str::FromStr, sync::mpsc, thread};
#[cfg(feature = "cookie")] #[cfg(feature = "cookie")]
use coo_kie::{Cookie, CookieJar}; use coo_kie::{Cookie, CookieJar};
use crate::io::{Filter, Io}; #[cfg(feature = "ws")]
use crate::io::Filter;
use crate::io::Io;
#[cfg(feature = "ws")]
use crate::ws::{error::WsClientError, WsClient, WsConnection}; use crate::ws::{error::WsClientError, WsClient, WsConnection};
use crate::{rt::System, service::ServiceFactory}; use crate::{rt::System, service::ServiceFactory};
use crate::{time::Millis, time::Seconds, util::Bytes}; use crate::{time::Millis, time::Seconds, util::Bytes};
@ -333,11 +336,13 @@ impl TestServer {
response.body().limit(10_485_760).await response.body().limit(10_485_760).await
} }
#[cfg(feature = "ws")]
/// Connect to a websocket server /// Connect to a websocket server
pub async fn ws(&mut self) -> Result<WsConnection<impl Filter>, WsClientError> { pub async fn ws(&mut self) -> Result<WsConnection<impl Filter>, WsClientError> {
self.ws_at("/").await self.ws_at("/").await
} }
#[cfg(feature = "ws")]
/// Connect to websocket server at a given path /// Connect to websocket server at a given path
pub async fn ws_at( pub async fn ws_at(
&mut self, &mut self,
@ -352,7 +357,7 @@ impl TestServer {
.await .await
} }
#[cfg(feature = "openssl")] #[cfg(all(feature = "openssl", feature = "ws"))]
/// Connect to a websocket server /// Connect to a websocket server
pub async fn wss( pub async fn wss(
&mut self, &mut self,
@ -363,7 +368,7 @@ impl TestServer {
self.wss_at("/").await self.wss_at("/").await
} }
#[cfg(feature = "openssl")] #[cfg(all(feature = "openssl", feature = "ws"))]
/// Connect to secure websocket server at a given path /// Connect to secure websocket server at a given path
pub async fn wss_at( pub async fn wss_at(
&mut self, &mut self,

View file

@ -31,6 +31,8 @@ pub use ntex_service::{forward_poll_ready, forward_poll_shutdown};
pub mod http; pub mod http;
pub mod web; pub mod web;
#[cfg(feature = "ws")]
pub mod ws; pub mod ws;
pub use self::service::{ pub use self::service::{

View file

@ -9,6 +9,7 @@ use crate::http::body::Body;
use crate::http::helpers::Writer; use crate::http::helpers::Writer;
use crate::http::{self, header, StatusCode}; use crate::http::{self, header, StatusCode};
use crate::util::{timeout::TimeoutError, BytesMut}; use crate::util::{timeout::TimeoutError, BytesMut};
#[cfg(feature = "ws")]
use crate::ws::error::HandshakeError; use crate::ws::error::HandshakeError;
use super::error::{self, ErrorContainer, ErrorRenderer, WebResponseError}; use super::error::{self, ErrorContainer, ErrorRenderer, WebResponseError};
@ -234,6 +235,7 @@ impl WebResponseError<DefaultError> for http::client::error::SendRequestError {
} }
} }
#[cfg(feature = "ws")]
/// Error renderer for ws::HandshakeError /// Error renderer for ws::HandshakeError
impl WebResponseError<DefaultError> for HandshakeError { impl WebResponseError<DefaultError> for HandshakeError {
fn error_response(&self, _: &HttpRequest) -> HttpResponse { fn error_response(&self, _: &HttpRequest) -> HttpResponse {

View file

@ -85,6 +85,8 @@ mod service;
pub mod test; pub mod test;
pub mod types; pub mod types;
mod util; mod util;
#[cfg(feature = "ws")]
pub mod ws; pub mod ws;
// re-export proc macro // re-export proc macro

View file

@ -11,14 +11,17 @@ use crate::http::error::{HttpError, PayloadError, ResponseError};
use crate::http::header::{HeaderName, HeaderValue, CONTENT_TYPE}; use crate::http::header::{HeaderName, HeaderValue, CONTENT_TYPE};
use crate::http::test::TestRequest as HttpTestRequest; use crate::http::test::TestRequest as HttpTestRequest;
use crate::http::{HttpService, Method, Payload, Request, StatusCode, Uri, Version}; use crate::http::{HttpService, Method, Payload, Request, StatusCode, Uri, Version};
#[cfg(feature = "ws")]
use crate::io::Sealed;
use crate::router::{Path, ResourceDef}; use crate::router::{Path, ResourceDef};
use crate::service::{ use crate::service::{
map_config, IntoService, IntoServiceFactory, Pipeline, Service, ServiceFactory, map_config, IntoService, IntoServiceFactory, Pipeline, Service, ServiceFactory,
}; };
use crate::time::{sleep, Millis, Seconds}; use crate::time::{sleep, Millis, Seconds};
use crate::util::{stream_recv, Bytes, BytesMut, Extensions, Ready, Stream}; use crate::util::{stream_recv, Bytes, BytesMut, Extensions, Ready, Stream};
#[cfg(feature = "ws")]
use crate::ws::{error::WsClientError, WsClient, WsConnection}; use crate::ws::{error::WsClientError, WsClient, WsConnection};
use crate::{io::Sealed, rt::System, server::Server}; use crate::{rt::System, server::Server};
use crate::web::error::{DefaultError, ErrorRenderer}; use crate::web::error::{DefaultError, ErrorRenderer};
use crate::web::httprequest::{HttpRequest, HttpRequestPool}; use crate::web::httprequest::{HttpRequest, HttpRequestPool};
@ -907,6 +910,7 @@ impl TestServer {
response.body().limit(10_485_760).await response.body().limit(10_485_760).await
} }
#[cfg(feature = "ws")]
/// Connect to websocket server at a given path /// Connect to websocket server at a given path
pub async fn ws_at(&self, path: &str) -> Result<WsConnection<Sealed>, WsClientError> { pub async fn ws_at(&self, path: &str) -> Result<WsConnection<Sealed>, WsClientError> {
if self.ssl { if self.ssl {
@ -947,6 +951,7 @@ impl TestServer {
} }
} }
#[cfg(feature = "ws")]
/// Connect to a websocket server /// Connect to a websocket server
pub async fn ws(&self) -> Result<WsConnection<Sealed>, WsClientError> { pub async fn ws(&self) -> Result<WsConnection<Sealed>, WsClientError> {
self.ws_at("/").await self.ws_at("/").await