1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
use futures_lite::{FutureExt, StreamExt};
use pharos::{Observable, ObserveConfig};
use ws_stream_wasm::{WsEvent, WsMessage, WsMeta, WsStream};

use crate::{sink_ext::SinkExt, Error};

/// A websocket connection for ws_stream_wasm
#[cfg_attr(docsrs, doc(cfg(feature = "ws_stream_wasm")))]
pub struct Connection {
    messages: WsStream,
    event_stream: pharos::Events<WsEvent>,
    meta: WsMeta,
}

impl Connection {
    /// Creates a new Connection from a WsMeta & WsTream combo
    pub async fn new((mut meta, messages): (WsMeta, WsStream)) -> Self {
        let event_stream = meta.observe(ObserveConfig::default()).await.unwrap();

        Connection {
            messages,
            event_stream,
            meta,
        }
    }
}

impl crate::next::Connection for Connection {
    async fn receive(&mut self) -> Option<crate::next::Message> {
        use crate::next::Message;
        loop {
            match self.next().await? {
                EventOrMessage::Event(WsEvent::Closed(close)) => {
                    return Some(Message::Close {
                        code: Some(close.code),
                        reason: Some(close.reason),
                    });
                }
                EventOrMessage::Event(WsEvent::Error | WsEvent::WsErr(_)) => {
                    return None;
                }
                EventOrMessage::Event(WsEvent::Open | WsEvent::Closing) => {
                    continue;
                }

                EventOrMessage::Message(WsMessage::Text(text)) => return Some(Message::Text(text)),

                EventOrMessage::Message(WsMessage::Binary(_)) => {
                    // We shouldn't receive binary messages, but ignore them if we do
                    continue;
                }
            }
        }
    }

    async fn send(&mut self, message: crate::next::Message) -> Result<(), Error> {
        use crate::next::Message;

        match message {
            Message::Text(text) => self.messages.send(WsMessage::Text(text)).await,
            Message::Close { code, reason } => match (code, reason) {
                (Some(code), Some(reason)) => self.meta.close_reason(code, reason).await,
                (Some(code), _) => self.meta.close_code(code).await,
                _ => self.meta.close().await,
            }
            .map(|_| ()),
            Message::Ping | Message::Pong => return Ok(()),
        }
        .map_err(|error| Error::Send(error.to_string()))
    }
}

impl Connection {
    async fn next(&mut self) -> Option<EventOrMessage> {
        let event = async { self.event_stream.next().await.map(EventOrMessage::Event) };
        let message = async { self.messages.next().await.map(EventOrMessage::Message) };

        event.race(message).await
    }
}

enum EventOrMessage {
    Event(WsEvent),
    Message(WsMessage),
}