discord-rpc-client/src/models/message.rs

96 lines
2.2 KiB
Rust

use std::io::{self, Read, Write};
use byteorder::{LittleEndian, ReadBytesExt, WriteBytesExt};
use serde::Serialize;
use serde_json;
use crate::error::{Error, Result};
#[derive(Debug, Copy, Clone, PartialEq)]
pub enum OpCode {
Handshake,
Frame,
Close,
Ping,
Pong,
}
impl TryFrom<u32> for OpCode {
type Error = Error;
fn try_from(int: u32) -> Result<Self, Self::Error> {
match int {
0 => Ok(OpCode::Handshake),
1 => Ok(OpCode::Frame),
2 => Ok(OpCode::Close),
3 => Ok(OpCode::Ping),
4 => Ok(OpCode::Pong),
_ => Err(Error::Conversion),
}
}
}
#[derive(Debug, PartialEq)]
pub struct Message {
pub opcode: OpCode,
pub payload: String,
}
impl Message {
pub fn new<T>(opcode: OpCode, payload: T) -> Self
where
T: Serialize,
{
Self {
opcode,
payload: serde_json::to_string(&payload).unwrap(),
}
}
pub fn encode(&self) -> Result<Vec<u8>> {
let mut bytes: Vec<u8> = vec![];
bytes.write_u32::<LittleEndian>(self.opcode as u32)?;
bytes.write_u32::<LittleEndian>(self.payload.len() as u32)?;
write!(bytes, "{}", self.payload)?;
Ok(bytes)
}
pub fn decode(bytes: &[u8]) -> Result<Self> {
let mut reader = io::Cursor::new(bytes);
let mut payload = String::new();
let opcode = OpCode::try_from(reader.read_u32::<LittleEndian>()?)?;
reader.read_u32::<LittleEndian>()?;
reader.read_to_string(&mut payload)?;
Ok(Self { opcode, payload })
}
}
#[cfg(test)]
mod tests {
use super::*;
#[derive(Debug, PartialEq, Serialize, Deserialize)]
struct Something {
empty: bool,
}
#[test]
fn test_encoder() {
let msg = Message::new(OpCode::Frame, Something { empty: true });
let encoded = msg.encode().unwrap();
let decoded = Message::decode(&encoded).unwrap();
assert_eq!(msg, decoded);
}
#[test]
fn test_opcode() {
assert_eq!(OpCode::try_from(0).ok(), Some(OpCode::Handshake));
assert_eq!(OpCode::try_from(4).ok(), Some(OpCode::Pong));
assert_eq!(OpCode::try_from(5).ok(), None);
}
}