aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/dav/caldecoder.rs8
-rw-r--r--src/dav/decoder.rs200
-rw-r--r--src/dav/error.rs1
-rw-r--r--src/dav/realization.rs6
-rw-r--r--src/dav/xml.rs23
5 files changed, 68 insertions, 170 deletions
diff --git a/src/dav/caldecoder.rs b/src/dav/caldecoder.rs
index 75af4b7..b45d649 100644
--- a/src/dav/caldecoder.rs
+++ b/src/dav/caldecoder.rs
@@ -7,25 +7,25 @@ use super::error;
// ---- EXTENSIONS ---
impl xml::QRead<Violation> for Violation {
- async fn qread(&self, xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
+ async fn qread(xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
unreachable!();
}
}
impl xml::QRead<Property> for Property {
- async fn qread(&self, xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
+ async fn qread(xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
unreachable!();
}
}
impl xml::QRead<PropertyRequest> for PropertyRequest {
- async fn qread(&self, xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
+ async fn qread(xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
unreachable!();
}
}
impl xml::QRead<ResourceType> for ResourceType {
- async fn qread(&self, xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
+ async fn qread(xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
unreachable!();
}
}
diff --git a/src/dav/decoder.rs b/src/dav/decoder.rs
index 1756464..a7fdca5 100644
--- a/src/dav/decoder.rs
+++ b/src/dav/decoder.rs
@@ -8,135 +8,25 @@ use quick_xml::reader::NsReader;
use tokio::io::AsyncBufRead;
use super::types::*;
-use super::error::*;
+use super::error::ParsingError;
+use super::xml::{QRead, Reader, IRead, DAV_URN, CAL_URN};
-/*
-// --- Traits ----
-
-trait Reader = AsyncBufRead+Unpin+'static;
-
-trait Decodable: Extension {
- async fn decode_propreq(xml: &mut PeekRead<impl Reader>) -> Result<Option<Self::PropertyRequest>, ParsingError>;
-}
-impl Decodable for NoExtension {
- async fn decode_propreq(xml: &mut PeekRead<impl Reader>) -> Result<Option<Disabled>, ParsingError> {
- Ok(None)
- }
-}
-
-pub trait QReadable<T: Reader>: Sized {
- async fn read(xml: &mut PeekRead<T>) -> Result<Self, ParsingError>;
-}
-
-// --- Peek read with namespaces
-
-const DAV_URN: &[u8] = b"DAV:";
-const CALDAV_URN: &[u8] = b"urn:ietf:params:xml:ns:caldav";
-const CARDDAV_URN: &[u8] = b"urn:ietf:params:xml:ns:carddav";
-//const XML_URN: &[u8] = b"xml";
-pub struct PeekRead<T: Reader> {
- evt: Event<'static>,
- rdr: NsReader<T>,
- buf: Vec<u8>,
-}
-impl<T: Reader> PeekRead<T> {
- async fn new(mut rdr: NsReader<T>) -> Result<Self, ParsingError> {
- let mut buf: Vec<u8> = vec![];
- let evt = rdr.read_event_into_async(&mut buf).await?.into_owned();
- buf.clear();
- Ok(Self { evt, rdr, buf })
- }
-
- fn peek(&self) -> &Event<'static> {
- &self.evt
- }
-
- /// skip tag. Can't skip end, can't skip eof.
- async fn skip(&mut self) -> Result<Event<'static>, ParsingError> {
- match &self.evt {
- Event::Start(b) => {
- let _span = self.rdr.read_to_end_into_async(b.to_end().name(), &mut self.buf).await?;
- self.next().await
- },
- Event::End(_) => Err(ParsingError::WrongToken),
- Event::Eof => Err(ParsingError::Eof),
- _ => self.next().await,
- }
- }
-
- /// read one more tag
- async fn next(&mut self) -> Result<Event<'static>, ParsingError> {
- let evt = self.rdr.read_event_into_async(&mut self.buf).await?.into_owned();
- self.buf.clear();
- let old_evt = std::mem::replace(&mut self.evt, evt);
- Ok(old_evt)
- }
-
-
- /// check if this is the desired tag
- fn is_tag(&self, ns: &[u8], key: &str) -> bool {
- let qname = match self.peek() {
- Event::Start(bs) | Event::Empty(bs) => bs.name(),
- Event::End(be) => be.name(),
- _ => return false,
- };
-
- let (extr_ns, local) = self.rdr.resolve_element(qname);
-
- if local.into_inner() != key.as_bytes() {
- return false
- }
-
- match extr_ns {
- ResolveResult::Bound(v) => v.into_inner() == ns,
- _ => false,
- }
- }
-
- /// find start tag
- async fn tag_start(&mut self, ns: &[u8], key: &str) -> Result<Event<'static>, ParsingError> {
- loop {
- match self.peek() {
- Event::Start(b) if self.is_tag(ns, key) => break,
- _ => { self.skip().await?; },
- }
- }
- self.next().await
- }
-
- // find stop tag
- async fn tag_stop(&mut self, ns: &[u8], key: &str) -> Result<Event<'static>, ParsingError> {
- loop {
- match self.peek() {
- Event::End(b) if self.is_tag(ns, key) => break,
- _ => { self.skip().await?; },
- }
- }
- self.next().await
- }
-}
-
-// ----- Decode ----
-
-impl<E: Decodable, T: Reader> QReadable<T> for PropFind<E> {
- async fn read(xml: &mut PeekRead<T>) -> Result<PropFind<E>, ParsingError> {
+impl<E: Extension> QRead<PropFind<E>> for PropFind<E> {
+ async fn qread(xml: &mut Reader<impl IRead>) -> Result<Option<Self>, ParsingError> {
// Find propfind
xml.tag_start(DAV_URN, "propfind").await?;
-
// Find any tag
let propfind: PropFind<E> = loop {
match xml.peek() {
Event::Start(_) if xml.is_tag(DAV_URN, "allprop") => {
xml.tag_start(DAV_URN, "allprop").await?;
- let r = PropFind::AllProp(Some(Include::read(xml).await?));
+ let r = PropFind::AllProp(Include::qread(xml).await?);
xml.tag_stop(DAV_URN, "allprop").await?;
break r
},
Event::Start(_) if xml.is_tag(DAV_URN, "prop") => {
- xml.tag_start(DAV_URN, "prop").await?;
- let r = PropFind::Prop(PropName::read(xml).await?);
- xml.tag_stop(DAV_URN, "prop").await?;
- break r
+ let propname = PropName::qread(xml).await?.ok_or(ParsingError::MissingChild)?;
+ break PropFind::Prop(propname);
},
Event::Empty(_) if xml.is_tag(DAV_URN, "allprop") => {
xml.next().await?;
@@ -153,49 +43,52 @@ impl<E: Decodable, T: Reader> QReadable<T> for PropFind<E> {
// Close tag
xml.tag_stop(DAV_URN, "propfind").await?;
- Ok(propfind)
+ Ok(Some(propfind))
}
}
-impl<E: Decodable, T: Reader> QReadable<T> for Include<E> {
- async fn read(xml: &mut PeekRead<T>) -> Result<Include<E>, ParsingError> {
+impl<E: Extension> QRead<Include<E>> for Include<E> {
+ async fn qread(xml: &mut Reader<impl IRead>) -> Result<Option<Self>, ParsingError> {
xml.tag_start(DAV_URN, "include").await?;
let mut acc: Vec<PropertyRequest<E>> = Vec::new();
loop {
match xml.peek() {
- Event::Start(_) | Event::Empty(_) => acc.push(PropertyRequest::read(xml).await?),
+ Event::Start(_) | Event::Empty(_) => {
+ PropertyRequest::qread(xml).await?.map(|v| acc.push(v));
+ },
Event::End(_) if xml.is_tag(DAV_URN, "include") => break,
_ => { xml.skip().await?; },
}
}
xml.tag_stop(DAV_URN, "include").await?;
- Ok(Include(acc))
+ Ok(Some(Include(acc)))
}
}
-impl<E: Decodable, T: Reader> QReadable<T> for PropName<E> {
- async fn read(xml: &mut PeekRead<T>) -> Result<PropName<E>, ParsingError> {
+impl<E: Extension> QRead<PropName<E>> for PropName<E> {
+ async fn qread(xml: &mut Reader<impl IRead>) -> Result<Option<Self>, ParsingError> {
xml.tag_start(DAV_URN, "prop").await?;
let mut acc: Vec<PropertyRequest<E>> = Vec::new();
loop {
match xml.peek() {
- Event::Start(_) | Event::Empty(_) => acc.push(PropertyRequest::read(xml).await?),
+ Event::Start(_) | Event::Empty(_) => {
+ PropertyRequest::qread(xml).await?.map(|v| acc.push(v));
+ },
Event::End(_) if xml.is_tag(DAV_URN, "prop") => break,
_ => { xml.skip().await?; },
}
}
xml.tag_stop(DAV_URN, "prop").await?;
- Ok(PropName(acc))
+ Ok(Some(PropName(acc)))
}
}
-impl<E: Decodable, T: Reader> QReadable<T> for PropertyRequest<E> {
- async fn read(xml: &mut PeekRead<T>) -> Result<PropertyRequest<E>, ParsingError> {
+impl<E: Extension> QRead<PropertyRequest<E>> for PropertyRequest<E> {
+ async fn qread(xml: &mut Reader<impl IRead>) -> Result<Option<Self>, ParsingError> {
loop {
- let (need_close, bs) = match xml.peek() {
- Event::Start(b) => (true, b),
- Event::Empty(b) => (false, b),
+ let bs = match xml.peek() {
+ Event::Start(b) | Event::Empty(b) => b,
_ => {
xml.skip().await?;
continue
@@ -212,6 +105,7 @@ impl<E: Decodable, T: Reader> QReadable<T> for PropertyRequest<E> {
b"displayname" => Some(PropertyRequest::DisplayName),
b"getcontentlanguage" => Some(PropertyRequest::GetContentLanguage),
b"getcontentlength" => Some(PropertyRequest::GetContentLength),
+ b"getcontenttype" => Some(PropertyRequest::GetContentType),
b"getetag" => Some(PropertyRequest::GetEtag),
b"getlastmodified" => Some(PropertyRequest::GetLastModified),
b"lockdiscovery" => Some(PropertyRequest::LockDiscovery),
@@ -223,18 +117,13 @@ impl<E: Decodable, T: Reader> QReadable<T> for PropertyRequest<E> {
// Option 2: an extension property
if maybe_res.is_none() {
- maybe_res = E::decode_propreq(xml).await?.map(PropertyRequest::Extension);
+ maybe_res = E::PropertyRequest::qread(xml).await?.map(PropertyRequest::Extension);
}
- // In any cases, we must close the opened tag
- if need_close {
- xml.skip().await?;
- }
+ // Close the current tag
+ xml.skip().await?;
- // Return if something is found - otherwise loop
- if let Some(res) = maybe_res {
- return Ok(res)
- }
+ return Ok(maybe_res)
}
}
}
@@ -242,6 +131,7 @@ impl<E: Decodable, T: Reader> QReadable<T> for PropertyRequest<E> {
#[cfg(test)]
mod tests {
use super::*;
+ use crate::dav::realization::Core;
#[tokio::test]
async fn basic_propfind_propname() {
@@ -253,11 +143,12 @@ mod tests {
</D:propfind>
"#;
- let mut rdr = PeekRead::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
- let got = PropFind::<NoExtension>::read(&mut rdr).await.unwrap();
- assert!(matches!(got, PropFind::PropName));
+ let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
+ let got = PropFind::<Core>::qread(&mut rdr).await.unwrap().unwrap();
+
+ assert_eq!(got, PropFind::<Core>::PropName);
}
-/*
+
#[tokio::test]
async fn basic_propfind_prop() {
let src = r#"<?xml version="1.0" encoding="utf-8" ?>
@@ -265,19 +156,20 @@ mod tests {
<garbage><old/></garbage>
<D:propfind xmlns:D="DAV:">
<D:prop>
- <displayname/>
- <getcontentlength/>
- <getcontenttype/>
- <getetag/>
- <getlastmodified/>
- <resourcetype/>
- <supportedlock/>
+ <D:displayname/>
+ <D:getcontentlength/>
+ <D:getcontenttype/>
+ <D:getetag/>
+ <D:getlastmodified/>
+ <D:resourcetype/>
+ <D:supportedlock/>
</D:prop>
</D:propfind>
"#;
- let mut rdr = PeekRead::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
- let got = PropFind::<NoExtension>::read(&mut rdr).await.unwrap();
+ let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
+ let got = PropFind::<Core>::qread(&mut rdr).await.unwrap().unwrap();
+
assert_eq!(got, PropFind::Prop(PropName(vec![
PropertyRequest::DisplayName,
PropertyRequest::GetContentLength,
@@ -288,6 +180,4 @@ mod tests {
PropertyRequest::SupportedLock,
])));
}
- */
}
-*/
diff --git a/src/dav/error.rs b/src/dav/error.rs
index 1db2895..5bd8ed3 100644
--- a/src/dav/error.rs
+++ b/src/dav/error.rs
@@ -2,6 +2,7 @@ use quick_xml::events::attributes::AttrError;
#[derive(Debug)]
pub enum ParsingError {
+ MissingChild,
NamespacePrefixAlreadyUsed,
WrongToken,
TagNotFound,
diff --git a/src/dav/realization.rs b/src/dav/realization.rs
index a02de94..1898173 100644
--- a/src/dav/realization.rs
+++ b/src/dav/realization.rs
@@ -6,8 +6,8 @@ use super::error;
#[derive(Debug, PartialEq)]
pub struct Disabled(());
impl xml::QRead<Disabled> for Disabled {
- async fn qread(&self, xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
- unreachable!();
+ async fn qread(xml: &mut xml::Reader<impl xml::IRead>) -> Result<Option<Self>, error::ParsingError> {
+ Ok(None)
}
}
impl xml::QWrite for Disabled {
@@ -20,6 +20,7 @@ impl xml::QWrite for Disabled {
///
/// Any extension is kooh is disabled through an object we can't build
/// due to a private inner element.
+#[derive(Debug, PartialEq)]
pub struct Core {}
impl dav::Extension for Core {
type Error = Disabled;
@@ -29,6 +30,7 @@ impl dav::Extension for Core {
}
// WebDAV with the base Calendar implementation (RFC4791)
+#[derive(Debug, PartialEq)]
pub struct Calendar {}
impl dav::Extension for Calendar
{
diff --git a/src/dav/xml.rs b/src/dav/xml.rs
index 495c9a5..5ebda02 100644
--- a/src/dav/xml.rs
+++ b/src/dav/xml.rs
@@ -5,6 +5,11 @@ use quick_xml::reader::NsReader;
use super::error::ParsingError;
+// Constants
+pub const DAV_URN: &[u8] = b"DAV:";
+pub const CAL_URN: &[u8] = b"urn:ietf:params:xml:ns:caldav";
+pub const CARD_URN: &[u8] = b"urn:ietf:params:xml:ns:carddav";
+
// Async traits
pub trait IWrite = AsyncWrite + Unpin;
pub trait IRead = AsyncBufRead + Unpin + 'static;
@@ -14,7 +19,7 @@ pub trait QWrite {
async fn qwrite(&self, xml: &mut Writer<impl IWrite>) -> Result<(), quick_xml::Error>;
}
pub trait QRead<T> {
- async fn qread(&self, xml: &mut Reader<impl IRead>) -> Result<Option<T>, ParsingError>;
+ async fn qread(xml: &mut Reader<impl IRead>) -> Result<Option<T>, ParsingError>;
}
/// Transform a Rust object into an XML stream of characters
@@ -42,24 +47,24 @@ impl<T: IWrite> Writer<T> {
/// Transform an XML stream of characters into a Rust object
pub struct Reader<T: IRead> {
+ pub rdr: NsReader<T>,
evt: Event<'static>,
- rdr: NsReader<T>,
buf: Vec<u8>,
}
impl<T: IRead> Reader<T> {
- async fn new(mut rdr: NsReader<T>) -> Result<Self, ParsingError> {
+ pub async fn new(mut rdr: NsReader<T>) -> Result<Self, ParsingError> {
let mut buf: Vec<u8> = vec![];
let evt = rdr.read_event_into_async(&mut buf).await?.into_owned();
buf.clear();
Ok(Self { evt, rdr, buf })
}
- fn peek(&self) -> &Event<'static> {
+ pub fn peek(&self) -> &Event<'static> {
&self.evt
}
/// skip tag. Can't skip end, can't skip eof.
- async fn skip(&mut self) -> Result<Event<'static>, ParsingError> {
+ pub async fn skip(&mut self) -> Result<Event<'static>, ParsingError> {
match &self.evt {
Event::Start(b) => {
let _span = self.rdr.read_to_end_into_async(b.to_end().name(), &mut self.buf).await?;
@@ -72,7 +77,7 @@ impl<T: IRead> Reader<T> {
}
/// read one more tag
- async fn next(&mut self) -> Result<Event<'static>, ParsingError> {
+ pub async fn next(&mut self) -> Result<Event<'static>, ParsingError> {
let evt = self.rdr.read_event_into_async(&mut self.buf).await?.into_owned();
self.buf.clear();
let old_evt = std::mem::replace(&mut self.evt, evt);
@@ -81,7 +86,7 @@ impl<T: IRead> Reader<T> {
/// check if this is the desired tag
- fn is_tag(&self, ns: &[u8], key: &str) -> bool {
+ pub fn is_tag(&self, ns: &[u8], key: &str) -> bool {
let qname = match self.peek() {
Event::Start(bs) | Event::Empty(bs) => bs.name(),
Event::End(be) => be.name(),
@@ -101,7 +106,7 @@ impl<T: IRead> Reader<T> {
}
/// find start tag
- async fn tag_start(&mut self, ns: &[u8], key: &str) -> Result<Event<'static>, ParsingError> {
+ pub async fn tag_start(&mut self, ns: &[u8], key: &str) -> Result<Event<'static>, ParsingError> {
loop {
match self.peek() {
Event::Start(b) if self.is_tag(ns, key) => break,
@@ -112,7 +117,7 @@ impl<T: IRead> Reader<T> {
}
// find stop tag
- async fn tag_stop(&mut self, ns: &[u8], key: &str) -> Result<Event<'static>, ParsingError> {
+ pub async fn tag_stop(&mut self, ns: &[u8], key: &str) -> Result<Event<'static>, ParsingError> {
loop {
match self.peek() {
Event::End(b) if self.is_tag(ns, key) => break,