feat(xmpp): push-based message parser

This commit is contained in:
Nikita Vilunov 2023-03-07 14:56:31 +01:00
parent 25c4d02ed2
commit d1dad72c08
5 changed files with 252 additions and 0 deletions

View File

@ -1 +1,203 @@
use quick_xml::events::Event;
use crate::prelude::*;
use crate::util::xml::*;
pub static XMLNS: &'static str = "jabber:client";
#[derive(PartialEq, Eq, Debug)]
pub struct Message {
from: Option<String>,
id: Option<String>,
to: Option<String>,
// default is Normal
r#type: MessageType,
lang: Option<String>,
subject: Option<String>,
body: String,
}
impl Message {
pub fn parse() -> impl Parser<Output = Result<Self>> {
MessageParser::Init
}
}
#[derive(Default)]
enum MessageParser {
#[default]
Init,
Outer(MessageParserState),
InSubject(MessageParserState),
InBody(MessageParserState),
}
#[derive(Default)]
struct MessageParserState {
from: Option<String>,
id: Option<String>,
to: Option<String>,
r#type: MessageType,
lang: Option<String>,
subject: Option<String>,
body: Option<String>,
}
impl Parser for MessageParser {
type Output = Result<Message>;
fn consume<'a>(self: Self, event: &Event<'a>) -> Continuation<Self, Self::Output> {
// TODO validate tag name and namespace at each stage
match self {
MessageParser::Init => {
if let Event::Start(ref bytes) = event {
let mut state: MessageParserState = Default::default();
for attr in bytes.attributes() {
let attr = fail_fast!(attr);
if attr.key.0 == b"from" {
let value = fail_fast!(std::str::from_utf8(&*attr.value));
state.from = Some(value.to_string())
} else if attr.key.0 == b"id" {
let value = fail_fast!(std::str::from_utf8(&*attr.value));
state.id = Some(value.to_string())
} else if attr.key.0 == b"to" {
let value = fail_fast!(std::str::from_utf8(&*attr.value));
state.to = Some(value.to_string())
}
}
Continuation::Continue(MessageParser::Outer(state))
} else {
Continuation::Final(Err(ffail!("Expected start")))
}
}
MessageParser::Outer(state) => {
match event {
Event::Start(ref bytes) => {
if bytes.name().0 == b"subject" {
Continuation::Continue(MessageParser::InSubject(state))
} else if bytes.name().0 == b"body" {
Continuation::Continue(MessageParser::InBody(state))
} else {
Continuation::Final(Err(ffail!("Unexpected XML tag")))
}
}
Event::End(_) => {
if let Some(body) = state.body {
Continuation::Final(Ok(Message {
from: state.from,
id: state.id,
to: state.to,
r#type: state.r#type,
lang: state.lang,
subject: state.subject,
body,
}))
} else {
Continuation::Final(Err(ffail!("Body not found")))
}
}
_ => {
Continuation::Final(Err(ffail!("Unexpected XML event: {event:?}")))
}
}
},
MessageParser::InSubject(mut state) => {
match event {
Event::Text(ref bytes) =>{
let subject = fail_fast!(std::str::from_utf8(&*bytes));
state.subject = Some(subject.to_string());
Continuation::Continue(MessageParser::InSubject(state))
}
Event::End(_) => {
Continuation::Continue(MessageParser::Outer(state))
}
_ => {
Continuation::Final(Err(ffail!("Unexpected XML event: {event:?}")))
}
}
}
MessageParser::InBody(mut state) => {
match event {
Event::Text(ref bytes) =>{
match std::str::from_utf8(&*bytes) {
Ok(subject) => {
state.body = Some(subject.to_string());
Continuation::Continue(MessageParser::InBody(state))
}
Err(err) => Continuation::Final(Err(err.into())),
}
}
Event::End(_) => {
Continuation::Continue(MessageParser::Outer(state))
}
_ => {
Continuation::Final(Err(ffail!("Unexpected XML event: {event:?}")))
}
}
}
}
}
}
#[derive(PartialEq, Eq, Debug)]
pub enum MessageType {
Chat,
Error,
Groupchat,
Headline,
Normal,
}
impl Default for MessageType {
fn default() -> Self {
MessageType::Normal
}
}
impl MessageType {
pub fn from_str(s: &str) -> Result<MessageType> {
use MessageType::*;
match s {
"chat" => Ok(Chat),
"error" => Ok(Error),
"groupchat" => Ok(Groupchat),
"headline" => Ok(Headline),
"normal" => Ok(Normal),
t => Err(ffail!("Unknown message type: {t}")),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use quick_xml::NsReader;
#[tokio::test]
async fn parse_message() {
let input = r#"<message id="aacea" type="chat" to="nikita@vlnv.dev"><subject>daa</subject><body>bbb</body></message>"#;
let mut reader = NsReader::from_reader(input.as_bytes());
let mut buf = vec![];
let event = reader.read_event_into_async(&mut buf).await.unwrap();
let mut parser = Message::parse().consume(&event);
let result = loop {
match parser {
Continuation::Final(res) => break res,
Continuation::Continue(next) => {
parser = next.consume(&reader.read_event_into_async(&mut buf).await.unwrap())
}
}
}
.unwrap();
assert_eq!(
result,
Message {
from: None,
id: Some("aacea".to_string()),
to: Some("nikita@vlnv.dev".to_string()),
r#type: MessageType::Normal,
lang: None,
subject: Some("daa".to_string()),
body: "bbb".to_string(),
}
)
}
}

View File

@ -2,6 +2,7 @@ pub mod client;
pub mod sasl;
pub mod stream;
pub mod tls;
pub mod stanzaerror;
// Implemented as a macro instead of a fn due to borrowck limitations
macro_rules! skip_text {

View File

@ -0,0 +1,25 @@
pub enum StanzaError {
BadRequest,
Conflict,
FeatureNotImplemented,
Forbidden,
Gone(String),
InternalServerError,
ItemNotFound,
JidMalformed,
NotAcceptable,
NotAllowed,
NotAuthorized,
PaymentRequired,
PolicyViolation,
RecipientUnavailable,
Redirect(String),
RegistrationRequired,
RemoteServerNotFound,
RemoteServerTimeout,
ResourceConstraint,
ServiceUnavailable,
SubscriptionRequired,
UndefinedCondition,
UnexpectedRequest,
}

View File

@ -3,6 +3,7 @@ use crate::prelude::*;
pub mod http;
pub mod table;
pub mod telemetry;
pub mod xml;
#[cfg(test)]
pub mod testkit;

23
src/util/xml.rs Normal file
View File

@ -0,0 +1,23 @@
use quick_xml::events::Event;
pub trait Parser: Sized {
type Output;
fn consume<'a>(self: Self, event: &Event<'a>) -> Continuation<Self, Self::Output>;
}
pub enum Continuation<Parser, Res> {
Final(Res),
Continue(Parser),
}
macro_rules! fail_fast {
($errorable: expr) => {
match $errorable {
Ok(i) => i,
Err(e) => return Continuation::Final(Err(e.into()))
}
};
}
pub(crate) use fail_fast;