lupt/src/ws_sansad.rs

154 lines
4.0 KiB
Rust

//! Ws Sansad manage websocket of each client
use actix::{Actor, Addr, Handler, Message, Running, StreamHandler};
use actix_web::web;
use actix_web_actors::ws;
use serde_json::{json, Value};
use crate::chat_pinnd::ChatPinnd;
use crate::chat_pinnd as pd;
pub struct WsSansad {
name: String,
isthiti: Isthiti,
pinnd: web::Data<Addr<ChatPinnd>>,
addr: Option<Addr<Self>>
}
#[allow(dead_code)]
enum Isthiti {
None,
Grih(Grih),
// VraktigatWaitlist
}
pub struct Grih {
kunjika: i32,
// name: String
}
// Handler Messages
pub struct Text {
pub text: String,
pub sender: String
}
pub struct SelfAddr(pub Addr<WsSansad>);
impl Actor for WsSansad {
type Context = ws::WebsocketContext<Self>;
fn stopping(&mut self, _: &mut Self::Context) -> Running {
futures::executor::block_on(self.end());
Running::Stop
}
}
impl StreamHandler<Result<ws::Message, ws::ProtocolError>> for WsSansad {
fn handle(&mut self, msg: Result<ws::Message, ws::ProtocolError>, ctx: &mut Self::Context) {
match msg {
Ok(ws::Message::Ping(msg)) => ctx.ping(&msg),
Ok(ws::Message::Text(msg)) => {
futures::executor::block_on(self.parse_text_handle(msg));
},
Ok(ws::Message::Close(msg)) => {
ctx.close(msg);
}
_ => ctx.close(None)
}
}
}
impl Handler<Text> for WsSansad {
type Result = ();
fn handle(&mut self, msg: Text, ctx: &mut Self::Context) -> Self::Result {
let json = json!({
"cmd": "text",
"text": msg.text,
"sender": msg.sender
});
ctx.text(json.to_string());
}
}
impl Handler<SelfAddr> for WsSansad {
type Result = ();
fn handle(&mut self, msg: SelfAddr, _: &mut Self::Context) -> Self::Result {
self.addr = Some(msg.0);
}
}
impl WsSansad {
pub fn new(pinnd: web::Data<Addr<ChatPinnd>>) -> Self {
WsSansad {
name: "()".to_owned(),
isthiti: Isthiti::None,
pinnd,
addr: None
}
}
async fn parse_text_handle(&mut self, msg: String) {
if let Ok(val) = serde_json::from_str::<Value>(&msg) {
match val.get("cmd").unwrap().as_str().unwrap() {
"name" => { self.name(val).await },
"join" => { self.join(val).await },
"text" => { self.text(val).await },
"end" => { self.end().await },
_ => ()
}
}
}
async fn name(&mut self, val: Value) {
self.name = val.get("name").unwrap().as_str().unwrap().to_owned();
}
async fn text(&mut self, val: Value) {
let text = val.get("text").unwrap().as_str().unwrap().to_owned();
let grih_kunjika = match &self.isthiti {
Isthiti::Grih(g) => {
g.kunjika
}, Isthiti::None => {
return;
}
};
self.pinnd.do_send(pd::Text {
grih_kunjika,
sender_name: self.name.clone(),
text
});
}
async fn join(&mut self, val: Value) {
let name = val.get("name").unwrap().as_str().unwrap().to_owned();
let length: Option<usize> = match val.get("length") {
Some(val) => Some(val.as_i64().unwrap() as usize),
None => None
};
let kunjika = self.pinnd.send(pd::Join{
grih: pd::JoinType::Name(name.clone()),
length,
addr: self.addr.clone().unwrap()
}).await.unwrap().unwrap();
self.isthiti = Isthiti::Grih(Grih {
kunjika,
// name
})
}
async fn end(&mut self) {
if let Isthiti::Grih(val) = &mut self.isthiti {
self.pinnd.do_send(pd::Delete {
grih_kunjika: val.kunjika,
addr: self.addr.clone().unwrap()
});
}
}
}
impl Message for Text { type Result = (); }
impl Message for SelfAddr { type Result = (); }