1
0
mirror of https://github.com/actix/examples synced 2025-01-23 14:24:35 +01:00

130 lines
2.9 KiB
Rust
Raw Normal View History

2018-05-11 14:35:43 -07:00
#![allow(dead_code)]
2019-12-16 13:09:54 +06:00
use std::io;
use actix::prelude::*;
use actix_codec::{Decoder, Encoder};
2018-05-11 14:35:43 -07:00
use byteorder::{BigEndian, ByteOrder};
use bytes::{BufMut, BytesMut};
use serde::{Deserialize, Serialize};
2018-05-11 14:35:43 -07:00
use serde_json as json;
/// Client request
#[derive(Serialize, Deserialize, Debug, Message)]
2019-12-16 13:09:54 +06:00
#[rtype(result = "()")]
2018-05-11 14:35:43 -07:00
#[serde(tag = "cmd", content = "data")]
pub enum ChatRequest {
/// List rooms
List,
/// Join rooms
Join(String),
/// Send message
Message(String),
/// Ping
Ping,
}
/// Server response
#[derive(Serialize, Deserialize, Debug, Message)]
2019-12-16 13:09:54 +06:00
#[rtype(result = "()")]
2018-05-11 14:35:43 -07:00
#[serde(tag = "cmd", content = "data")]
pub enum ChatResponse {
Ping,
/// List of rooms
Rooms(Vec<String>),
/// Joined
Joined(String),
/// Message
Message(String),
}
/// Codec for Client -> Server transport
pub struct ChatCodec;
impl Decoder for ChatCodec {
type Item = ChatRequest;
type Error = io::Error;
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
let size = {
if src.len() < 2 {
return Ok(None);
}
BigEndian::read_u16(src.as_ref()) as usize
};
if src.len() >= size + 2 {
2019-12-16 13:09:54 +06:00
let _ = src.split_to(2);
2018-05-11 14:35:43 -07:00
let buf = src.split_to(size);
Ok(Some(json::from_slice::<ChatRequest>(&buf)?))
} else {
Ok(None)
}
}
}
2020-09-12 16:49:45 +01:00
impl Encoder<ChatResponse> for ChatCodec {
2018-05-11 14:35:43 -07:00
type Error = io::Error;
fn encode(
2019-03-09 18:03:09 -08:00
&mut self,
msg: ChatResponse,
dst: &mut BytesMut,
2018-05-11 14:35:43 -07:00
) -> Result<(), Self::Error> {
let msg = json::to_string(&msg).unwrap();
let msg_ref: &[u8] = msg.as_ref();
dst.reserve(msg_ref.len() + 2);
2019-12-16 13:09:54 +06:00
dst.put_u16(msg_ref.len() as u16);
2018-05-11 14:35:43 -07:00
dst.put(msg_ref);
Ok(())
}
}
/// Codec for Server -> Client transport
pub struct ClientChatCodec;
impl Decoder for ClientChatCodec {
type Item = ChatResponse;
type Error = io::Error;
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
let size = {
if src.len() < 2 {
return Ok(None);
}
BigEndian::read_u16(src.as_ref()) as usize
};
if src.len() >= size + 2 {
2019-12-16 13:09:54 +06:00
let _ = src.split_to(2);
2018-05-11 14:35:43 -07:00
let buf = src.split_to(size);
Ok(Some(json::from_slice::<ChatResponse>(&buf)?))
} else {
Ok(None)
}
}
}
2020-09-12 16:49:45 +01:00
impl Encoder<ChatRequest> for ClientChatCodec {
2018-05-11 14:35:43 -07:00
type Error = io::Error;
fn encode(
2019-03-09 18:03:09 -08:00
&mut self,
msg: ChatRequest,
dst: &mut BytesMut,
2018-05-11 14:35:43 -07:00
) -> Result<(), Self::Error> {
let msg = json::to_string(&msg).unwrap();
let msg_ref: &[u8] = msg.as_ref();
dst.reserve(msg_ref.len() + 2);
2019-12-16 13:09:54 +06:00
dst.put_u16(msg_ref.len() as u16);
2018-05-11 14:35:43 -07:00
dst.put(msg_ref);
Ok(())
}
}