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

Feature request: Non-blocking WebSocket::read #378

Closed
emilk opened this issue Sep 18, 2023 · 8 comments
Closed

Feature request: Non-blocking WebSocket::read #378

emilk opened this issue Sep 18, 2023 · 8 comments

Comments

@emilk
Copy link

emilk commented Sep 18, 2023

WebSocket::read is currently blocking. I would love a non-blocking version that tries to read a message, and if there is none, returns immediately.

I want to accomplish something like this:

let socket = …
loop {
    while let Some(msg) = socket.try_read() {
        handle(msg);
    }

    if some_user_input {
        socket.send();
    }}

I can accomplish this using tokio_tungstenite::connect_async and calling split on the resulting stream, but I rather not pull in async and tokio if I can avoid it!

@agalakhov
Copy link
Member

WebSocket::read is blocking if and only if the underlying stream is blocking too. It is non-blocking if used with non-blocking version of TcpStream.

@agalakhov
Copy link
Member

This feature is already there.

@agalakhov agalakhov closed this as not planned Won't fix, can't repro, duplicate, stale Sep 18, 2023
@emilk
Copy link
Author

emilk commented Oct 1, 2023

That's great news! But how do I accomplish this? Can I use tungstenite::connect or do I need to I need to use WebSocket::from_raw_socket? Is there an example I can follow?

@emilk
Copy link
Author

emilk commented Oct 1, 2023

I think I figured it out:

    match socket.get_mut() {
        tungstenite::stream::MaybeTlsStream::Plain(stream) => stream.set_nonblocking(true),
        tungstenite::stream::MaybeTlsStream::Rustls(stream) => {
            stream.get_mut().set_nonblocking(true)
        }
        _ => unimplemented!(),
    }?;

it would be nice if there was a socket.stream_mut() to access the underlying stream though

@agalakhov
Copy link
Member

@emilk
Copy link
Author

emilk commented Oct 2, 2023

Sorry, I meant the underlying TcpStream, instead of doing the incomplete match shown above

@agalakhov
Copy link
Member

The WebSocket object does not know anything about the underlying stream at all. The underlying stream may or may not support non-blocking mode. While tungstenite provides a convenience wrapper for TcpStream and TLS from either rustls or native-tls, other variants have different ones. For example, tokio-tungstenite replaces std's TcpStream with the tokio one, which is always non-blocking and does not have any methods to switch modes. That's why we can't make any assumptions about underlying TcpStream. All we know is that it is readable and writable.

@williamdes
Copy link

williamdes commented Sep 27, 2024

I think I figured it out:

    match socket.get_mut() {
        tungstenite::stream::MaybeTlsStream::Plain(stream) => stream.set_nonblocking(true),
        tungstenite::stream::MaybeTlsStream::Rustls(stream) => {
            stream.get_mut().set_nonblocking(true)
        }
        _ => unimplemented!(),
    }?;

it would be nice if there was a socket.stream_mut() to access the underlying stream though

This still works fine, but an easier path is to set a read timeout
As said on #113 (comment)

But it requires some filtering

             Err(err) => {
                match err {
                    // Silently drop the error: Processing error: IO error: Resource temporarily unavailable (os error 11)
                    // That occurs when no messages are to be read
                    Error::Io(ref e) if e.kind() == std::io::ErrorKind::WouldBlock => Ok(()),
                    _ => Err(err),
                }
            },

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

3 participants