How to wrap message handler? #378
-
I need to implement a middleware like functionality for message handlers. I think |
Beta Was this translation helpful? Give feedback.
Replies: 1 comment 1 reply
-
Hey ! If you need to keep dynamic parameters for your message handler you can manually implement use std::{marker::PhantomData, sync::Arc};
use socketioxide::{
adapter::Adapter,
extract::{Data, Extension, SocketRef},
handler::{FromMessageParts, MessageHandler},
socket::Socket,
SocketIo,
};
/// An extension wrapper for the example.
#[derive(Debug, Clone)]
struct Info(&'static str);
struct MessageMiddleware<H, A, T> {
handler: H,
_phantom: PhantomData<fn(A, T)>,
}
impl<H, A, T> MessageMiddleware<H, A, T> {
pub fn new(handler: H) -> Self {
MessageMiddleware {
handler,
_phantom: PhantomData::default(),
}
}
}
impl<H, A, T> MessageHandler<A, T> for MessageMiddleware<H, A, T>
where
H: MessageHandler<A, T>,
A: Adapter,
T: 'static,
{
fn call(
&self,
s: Arc<Socket<A>>,
mut v: serde_json::Value,
mut p: Vec<axum::body::Bytes>,
ack_id: Option<i64>,
) {
// We set an extension on the socket.
s.extensions.insert(Info("super test!"));
// We parse the incoming data to print it.
let data: Result<Data<String>, _> = Data::from_message_parts(&s, &mut v, &mut p, &ack_id);
match data {
Ok(Data(data)) => println!("received data: {:?}", data),
Err(err) => println!("deserialization error: {:?}", err),
};
// We forward the call to the inner handler
self.handler.call(s, v, p, ack_id);
}
}
fn my_first_event_handler(s: SocketRef, Data(msg): Data<String>, Extension(ext): Extension<Info>) {
s.emit("test", msg).unwrap();
assert!(matches!(ext, Info("super test!")));
}
fn my_second_event_handler(s: SocketRef, Extension(ext): Extension<Info>) {
println!("socket: {}, info: {:?}", s.id, ext);
assert!(matches!(ext, Info("super test!")));
}
#[tokio::main]
async fn main() {
let (layer, io) = SocketIo::new_layer();
io.ns("/", move |s: SocketRef| {
s.on("test_1", MessageMiddleware::new(my_first_event_handler));
s.on("test_2", MessageMiddleware::new(my_second_event_handler));
});
let app = axum::Router::new().layer(layer);
let listener = tokio::net::TcpListener::bind("0.0.0.0:3000").await.unwrap();
axum::serve(listener, app).await.unwrap();
} Note that in the middleware code, it is important to use the extractor |
Beta Was this translation helpful? Give feedback.
Hey ! If you need to keep dynamic parameters for your message handler you can manually implement
MessageHandler
for a middleware struct that wraps your handlers: