use crate::{irc::*, MaybeOwned, MaybeOwnedIndex, Validator};
#[derive(Debug, Clone, PartialEq)]
#[cfg_attr(feature = "serde", derive(::serde::Serialize, ::serde::Deserialize))]
pub enum HostTargetKind<'a> {
Start {
target: &'a str,
},
End,
}
#[derive(Clone, PartialEq)]
pub struct HostTarget<'a> {
raw: MaybeOwned<'a>,
source: MaybeOwnedIndex,
viewers: Option<usize>,
target: Option<MaybeOwnedIndex>,
}
impl<'a> HostTarget<'a> {
raw!();
str_field!(
source
);
pub fn viewers(&self) -> Option<usize> {
self.viewers
}
pub fn host_target_kind(&self) -> HostTargetKind<'_> {
match self.target {
Some(index) => HostTargetKind::Start {
target: &self.raw[index],
},
None => HostTargetKind::End,
}
}
}
impl<'a> FromIrcMessage<'a> for HostTarget<'a> {
type Error = MessageError;
fn from_irc(msg: IrcMessage<'a>) -> Result<Self, Self::Error> {
msg.expect_command(IrcMessage::HOST_TARGET)?;
let mut data = msg.expect_data()?.splitn(2, char::is_whitespace);
let target = match data.next() {
Some("-") => None,
Some(t) => {
let kind = msg.expect_data_index()?.resize(t.len());
Some(kind)
}
None => return Err(MessageError::ExpectedData),
};
let viewers = data.next().and_then(|s| s.parse().ok());
let this = Self {
source: msg.expect_arg_index(0)?,
viewers,
target,
raw: msg.raw,
};
Ok(this)
}
into_inner_raw!();
}
into_owned!(HostTarget {
raw,
source,
viewers,
target,
});
impl_custom_debug!(HostTarget {
raw,
source,
viewers,
host_target_kind,
});
serde_struct!(HostTarget {
raw,
source,
viewers,
host_target_kind
});
#[cfg(test)]
mod tests {
use super::*;
#[test]
#[cfg(feature = "serde")]
fn host_target_serde() {
let input = ":tmi.twitch.tv HOSTTARGET #shaken_bot :museun 1024\r\n";
crate::serde::round_trip_json::<HostTarget>(input);
crate::serde::round_trip_rmp::<HostTarget>(input);
}
#[test]
fn host_target_start() {
let input = ":tmi.twitch.tv HOSTTARGET #shaken_bot :museun 1024\r\n";
for msg in parse(input).map(|s| s.unwrap()) {
let ht = HostTarget::from_irc(msg).unwrap();
assert_eq!(ht.source(), "#shaken_bot");
assert_eq!(ht.viewers().unwrap(), 1024);
assert_eq!(
ht.host_target_kind(),
HostTargetKind::Start { target: "museun" }
);
}
}
#[test]
fn host_target_start_none() {
let input = ":tmi.twitch.tv HOSTTARGET #shaken_bot :museun -\r\n";
for msg in parse(input).map(|s| s.unwrap()) {
let ht = HostTarget::from_irc(msg).unwrap();
assert_eq!(ht.source(), "#shaken_bot");
assert!(ht.viewers().is_none());
assert_eq!(
ht.host_target_kind(),
HostTargetKind::Start { target: "museun" }
);
}
}
#[test]
fn host_target_end() {
let input = ":tmi.twitch.tv HOSTTARGET #shaken_bot :- 1024\r\n";
for msg in parse(input).map(|s| s.unwrap()) {
let ht = HostTarget::from_irc(msg).unwrap();
assert_eq!(ht.source(), "#shaken_bot");
assert_eq!(ht.viewers().unwrap(), 1024);
assert_eq!(ht.host_target_kind(), HostTargetKind::End);
}
}
}