20 releases (11 breaking)

0.12.0 Mar 20, 2024
0.10.2 Jan 22, 2024
0.9.1 Dec 21, 2023
0.7.2 Nov 22, 2023
0.3.0 Jun 17, 2023

#112 in WebSocket

Download history 1066/week @ 2023-12-31 1477/week @ 2024-01-07 1236/week @ 2024-01-14 1512/week @ 2024-01-21 1510/week @ 2024-01-28 1759/week @ 2024-02-04 1342/week @ 2024-02-11 1211/week @ 2024-02-18 1407/week @ 2024-02-25 1781/week @ 2024-03-03 1843/week @ 2024-03-10 1604/week @ 2024-03-17 1392/week @ 2024-03-24 1623/week @ 2024-03-31 1623/week @ 2024-04-07 1624/week @ 2024-04-14

6,404 downloads per month
Used in 5 crates (via socketioxide)

MIT license

155KB
3K SLoC

Engineioxide does the heavy lifting for socketioxide, a socket.io server implementation in Rust which integrates with the tower stack.

You can still use engineioxide as a standalone crate to talk with an engine.io client.

Supported Protocols

You can enable support for other engine.io protocol implementations through feature flags. The latest protocol version (v4) is enabled by default.

To add support for the v3 protocol version, adjust your dependency configuration accordingly:

[dependencies]
# Enables the `v3` protocol (`v4` is always enabled, as it's the default).
engineioxide = { version = "0.3.0", features = ["v3"] }

Feature flags :

  • v3: Enable the engine.io v3 protocol
  • tracing: Enable tracing logs with the tracing crate

Basic example with axum :

use engineioxide::layer::EngineIoLayer;
use engineioxide::handler::EngineIoHandler;
use engineioxide::{Socket, DisconnectReason};
use std::sync::atomic::{AtomicUsize, Ordering};
use std::sync::{Arc, Mutex};
use axum::routing::get;
// Global state, with axum it must be clonable
#[derive(Debug, Default, Clone)]
struct MyHandler {
    user_cnt: Arc<AtomicUsize>,
}

// Socket state
#[derive(Debug, Default)]
struct SocketState {
    id: Mutex<String>,
}

impl EngineIoHandler for MyHandler {
    type Data = SocketState;

    fn on_connect(&self, socket: Arc<Socket<SocketState>>) { 
        let cnt = self.user_cnt.fetch_add(1, Ordering::Relaxed) + 1;
        socket.emit(cnt.to_string()).ok();
    }
    fn on_disconnect(&self, socket: Arc<Socket<SocketState>>, reason: DisconnectReason) { 
        let cnt = self.user_cnt.fetch_sub(1, Ordering::Relaxed) - 1;
        socket.emit(cnt.to_string()).ok();
    }
    fn on_message(&self, msg: String, socket: Arc<Socket<SocketState>>) { 
        *socket.data.id.lock().unwrap() = msg; // bind a provided user id to a socket
    }
    fn on_binary(&self, data: Vec<u8>, socket: Arc<Socket<SocketState>>) { }
}

// Create a new engineio layer
let layer = EngineIoLayer::new(MyHandler::default());

let app = axum::Router::<()>::new()
    .route("/", get(|| async { "Hello, World!" }))
    .layer(layer);

// Spawn the axum server

Dependencies

~9–20MB
~256K SLoC