//! Messages to and from the server. #![experimental] use std::borrow::ToOwned; use std::str::FromStr; /// Represents something that can be converted to a message. pub trait ToMessage { /// Convert to message. fn to_message(&self) -> Message; } /// IRC Message data. #[experimental] #[deriving(Clone, PartialEq, Show)] pub struct Message { /// The message prefix (or source) as defined by [RFC 2812](http://tools.ietf.org/html/rfc2812). pub prefix: Option, /// The IRC command as defined by [RFC 2812](http://tools.ietf.org/html/rfc2812). pub command: String, /// The command arguments. pub args: Vec, /// The message suffix as defined by [RFC 2812](http://tools.ietf.org/html/rfc2812). /// This is the only part of the message that is allowed to contain spaces. pub suffix: Option, } impl Message { /// Creates a new Message. #[experimental] pub fn new(prefix: Option<&str>, command: &str, args: Option>, suffix: Option<&str>) -> Message { Message { prefix: prefix.map(|s| s.to_owned()), command: command.to_owned(), args: args.map_or(Vec::new(), |v| v.iter().map(|&s| s.to_owned()).collect()), suffix: suffix.map(|s| s.to_owned()), } } /// Converts a Message into a String according to the IRC protocol. #[experimental] pub fn into_string(&self) -> String { let mut ret = String::new(); if let Some(ref prefix) = self.prefix { ret.push(':'); ret.push_str(prefix[]); ret.push(' '); } ret.push_str(self.command[]); for arg in self.args.iter() { ret.push(' '); ret.push_str(arg[]); } if let Some(ref suffix) = self.suffix { ret.push_str(" :"); ret.push_str(suffix[]); } ret.push_str("\r\n"); ret } } impl ToMessage for Message { fn to_message(&self) -> Message { self.clone() } } impl FromStr for Message { fn from_str(s: &str) -> Option { let mut state = s.clone(); if s.len() == 0 { return None } let prefix = if state.starts_with(":") { let prefix = state.find(' ').map(|i| state[1..i]); state = state.find(' ').map_or("", |i| state[i+1..]); prefix } else { None }; let suffix = if state.contains(":") { let suffix = state.find(':').map(|i| state[i+1..state.len()-2]); state = state.find(':').map_or("", |i| state[..i]); suffix } else { None }; let command = match state.find(' ').map(|i| state[..i]) { Some(cmd) => { state = state.find(' ').map_or("", |i| state[i+1..]); cmd } _ => return None }; if suffix.is_none() { state = state[..state.len() - 2] } let args: Vec<_> = state.splitn(14, ' ').filter(|s| s.len() != 0).collect(); Some(Message::new(prefix, command, if args.len() > 0 { Some(args) } else { None }, suffix)) } } #[cfg(test)] mod test { use super::Message; #[test] fn new() { let message = Message { prefix: None, command: format!("PRIVMSG"), args: vec![format!("test")], suffix: Some(format!("Testing!")), }; assert_eq!(Message::new(None, "PRIVMSG", Some(vec!["test"]), Some("Testing!")), message); } #[test] fn into_string() { let message = Message { prefix: None, command: format!("PRIVMSG"), args: vec![format!("test")], suffix: Some(format!("Testing!")), }; assert_eq!(message.into_string()[], "PRIVMSG test :Testing!\r\n"); let message = Message { prefix: Some(format!("test!test@test")), command: format!("PRIVMSG"), args: vec![format!("test")], suffix: Some(format!("Still testing!")), }; assert_eq!(message.into_string()[], ":test!test@test PRIVMSG test :Still testing!\r\n"); } #[test] fn from_string() { let message = Message { prefix: None, command: format!("PRIVMSG"), args: vec![format!("test")], suffix: Some(format!("Testing!")), }; assert_eq!("PRIVMSG test :Testing!\r\n".parse(), Some(message)); let message = Message { prefix: Some(format!("test!test@test")), command: format!("PRIVMSG"), args: vec![format!("test")], suffix: Some(format!("Still testing!")), }; assert_eq!(":test!test@test PRIVMSG test :Still testing!\r\n".parse(), Some(message)); } }