2024-03-08 07:17:03 +00:00
|
|
|
use quick_xml::events::Event;
|
2024-03-08 08:55:33 +00:00
|
|
|
use chrono::DateTime;
|
2024-03-04 12:36:41 +00:00
|
|
|
|
|
|
|
use super::types::*;
|
2024-03-05 18:06:04 +00:00
|
|
|
use super::error::ParsingError;
|
2024-03-08 08:55:33 +00:00
|
|
|
use super::xml::{Node, QRead, Reader, IRead, DAV_URN};
|
2024-03-04 12:36:41 +00:00
|
|
|
|
2024-03-06 15:09:20 +00:00
|
|
|
//@TODO (1) Rewrite all objects as Href,
|
|
|
|
// where we return Ok(None) instead of trying to find the object at any cost.
|
|
|
|
// Add a xml.find<E: Qread>() -> Result<Option<E>, ParsingError> or similar for the cases we
|
|
|
|
// really need the object
|
|
|
|
// (2) Rewrite QRead and replace Result<Option<_>, _> with Result<_, _>, not found being a possible
|
|
|
|
// error.
|
|
|
|
// (3) Rewrite vectors with xml.collect<E: QRead>() -> Result<Vec<E>, _>
|
|
|
|
// (4) Something for alternatives would be great but no idea yet
|
|
|
|
|
2024-03-06 09:12:02 +00:00
|
|
|
// ---- ROOT ----
|
2024-03-06 11:42:27 +00:00
|
|
|
|
|
|
|
/// Propfind request
|
2024-03-05 18:06:04 +00:00
|
|
|
impl<E: Extension> QRead<PropFind<E>> for PropFind<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "propfind").await?;
|
2024-03-04 21:27:37 +00:00
|
|
|
let propfind: PropFind<E> = loop {
|
2024-03-07 11:25:22 +00:00
|
|
|
// allprop
|
|
|
|
if let Some(_) = xml.maybe_open(DAV_URN, "allprop").await? {
|
|
|
|
xml.close().await?;
|
2024-03-08 10:34:24 +00:00
|
|
|
let includ = xml.maybe_find::<Include<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
break PropFind::AllProp(includ)
|
2024-03-04 16:55:48 +00:00
|
|
|
}
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
// propname
|
|
|
|
if let Some(_) = xml.maybe_open(DAV_URN, "propname").await? {
|
|
|
|
xml.close().await?;
|
|
|
|
break PropFind::PropName
|
|
|
|
}
|
|
|
|
|
|
|
|
// prop
|
|
|
|
let (mut maybe_prop, mut dirty) = (None, false);
|
|
|
|
xml.maybe_read::<PropName<E>>(&mut maybe_prop, &mut dirty).await?;
|
|
|
|
if let Some(prop) = maybe_prop {
|
|
|
|
break PropFind::Prop(prop)
|
|
|
|
}
|
|
|
|
|
|
|
|
// not found, skipping
|
|
|
|
xml.skip().await?;
|
|
|
|
};
|
|
|
|
xml.close().await?;
|
2024-03-04 21:27:37 +00:00
|
|
|
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(propfind)
|
2024-03-04 21:27:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 11:42:27 +00:00
|
|
|
/// PROPPATCH request
|
|
|
|
impl<E: Extension> QRead<PropertyUpdate<E>> for PropertyUpdate<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "propertyupdate").await?;
|
|
|
|
let collected_items = xml.collect::<PropertyUpdateItem<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(PropertyUpdate(collected_items))
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 15:09:20 +00:00
|
|
|
/// Generic response
|
2024-03-06 19:58:41 +00:00
|
|
|
impl<E: Extension, N: Node<N>> QRead<Multistatus<E,N>> for Multistatus<E,N> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "multistatus").await?;
|
2024-03-06 17:35:54 +00:00
|
|
|
let mut responses = Vec::new();
|
|
|
|
let mut responsedescription = None;
|
|
|
|
|
|
|
|
loop {
|
2024-03-06 22:24:54 +00:00
|
|
|
let mut dirty = false;
|
|
|
|
xml.maybe_push(&mut responses, &mut dirty).await?;
|
|
|
|
xml.maybe_read(&mut responsedescription, &mut dirty).await?;
|
|
|
|
if !dirty {
|
2024-03-06 17:35:54 +00:00
|
|
|
match xml.peek() {
|
|
|
|
Event::End(_) => break,
|
|
|
|
_ => xml.skip().await?,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Multistatus { responses, responsedescription })
|
2024-03-06 17:35:54 +00:00
|
|
|
}
|
|
|
|
}
|
2024-03-06 11:42:27 +00:00
|
|
|
|
2024-03-06 15:09:20 +00:00
|
|
|
// LOCK REQUEST
|
|
|
|
impl QRead<LockInfo> for LockInfo {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "lockinfo").await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
let (mut m_scope, mut m_type, mut owner) = (None, None, None);
|
|
|
|
loop {
|
2024-03-06 22:24:54 +00:00
|
|
|
let mut dirty = false;
|
|
|
|
xml.maybe_read::<LockScope>(&mut m_scope, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<LockType>(&mut m_type, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Owner>(&mut owner, &mut dirty).await?;
|
|
|
|
|
|
|
|
if !dirty {
|
2024-03-06 15:09:20 +00:00
|
|
|
match xml.peek() {
|
|
|
|
Event::End(_) => break,
|
|
|
|
_ => xml.skip().await?,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
match (m_scope, m_type) {
|
2024-03-06 22:24:54 +00:00
|
|
|
(Some(lockscope), Some(locktype)) => Ok(LockInfo { lockscope, locktype, owner }),
|
2024-03-06 15:09:20 +00:00
|
|
|
_ => Err(ParsingError::MissingChild),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// LOCK RESPONSE
|
|
|
|
impl<E: Extension> QRead<PropValue<E>> for PropValue<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "prop").await?;
|
2024-03-08 08:55:33 +00:00
|
|
|
let acc = xml.collect::<Property<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(PropValue(acc))
|
2024-03-06 15:09:20 +00:00
|
|
|
}
|
|
|
|
}
|
2024-03-06 11:42:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
/// Error response
|
2024-03-06 09:12:02 +00:00
|
|
|
impl<E: Extension> QRead<Error<E>> for Error<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "error").await?;
|
|
|
|
let violations = xml.collect::<Violation<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Error(violations))
|
2024-03-06 09:12:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 11:42:27 +00:00
|
|
|
|
|
|
|
|
2024-03-06 09:12:02 +00:00
|
|
|
// ---- INNER XML
|
2024-03-06 19:58:41 +00:00
|
|
|
impl<E: Extension, N: Node<N>> QRead<Response<E,N>> for Response<E,N> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "response").await?;
|
2024-03-06 19:58:41 +00:00
|
|
|
let (mut status, mut error, mut responsedescription, mut location) = (None, None, None, None);
|
|
|
|
let mut href = Vec::new();
|
|
|
|
let mut propstat = Vec::new();
|
2024-03-06 17:35:54 +00:00
|
|
|
|
2024-03-06 19:58:41 +00:00
|
|
|
loop {
|
2024-03-06 22:24:54 +00:00
|
|
|
let mut dirty = false;
|
|
|
|
xml.maybe_read::<Status>(&mut status, &mut dirty).await?;
|
|
|
|
xml.maybe_push::<Href>(&mut href, &mut dirty).await?;
|
|
|
|
xml.maybe_push::<PropStat<E,N>>(&mut propstat, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Error<E>>(&mut error, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<ResponseDescription>(&mut responsedescription, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Location>(&mut location, &mut dirty).await?;
|
|
|
|
|
|
|
|
if !dirty {
|
2024-03-06 19:58:41 +00:00
|
|
|
match xml.peek() {
|
|
|
|
Event::End(_) => break,
|
|
|
|
_ => { xml.skip().await? },
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 19:58:41 +00:00
|
|
|
match (status, &propstat[..], &href[..]) {
|
2024-03-06 22:24:54 +00:00
|
|
|
(Some(status), &[], &[_, ..]) => Ok(Response {
|
2024-03-06 19:58:41 +00:00
|
|
|
status_or_propstat: StatusOrPropstat::Status(href, status),
|
|
|
|
error, responsedescription, location,
|
2024-03-06 22:24:54 +00:00
|
|
|
}),
|
|
|
|
(None, &[_, ..], &[_, ..]) => Ok(Response {
|
2024-03-06 19:58:41 +00:00
|
|
|
status_or_propstat: StatusOrPropstat::PropStat(href.into_iter().next().unwrap(), propstat),
|
|
|
|
error, responsedescription, location,
|
2024-03-06 22:24:54 +00:00
|
|
|
}),
|
2024-03-06 19:58:41 +00:00
|
|
|
(Some(_), &[_, ..], _) => Err(ParsingError::InvalidValue),
|
|
|
|
_ => Err(ParsingError::MissingChild),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<E: Extension, N: Node<N>> QRead<PropStat<E,N>> for PropStat<E,N> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "propstat").await?;
|
|
|
|
|
|
|
|
let (mut m_prop, mut m_status, mut error, mut responsedescription) = (None, None, None, None);
|
|
|
|
|
|
|
|
loop {
|
|
|
|
let mut dirty = false;
|
|
|
|
xml.maybe_read::<N>(&mut m_prop, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Status>(&mut m_status, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Error<E>>(&mut error, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<ResponseDescription>(&mut responsedescription, &mut dirty).await?;
|
|
|
|
|
|
|
|
if !dirty {
|
|
|
|
match xml.peek() {
|
|
|
|
Event::End(_) => break,
|
|
|
|
_ => xml.skip().await?,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
match (m_prop, m_status) {
|
|
|
|
(Some(prop), Some(status)) => Ok(PropStat { prop, status, error, responsedescription }),
|
|
|
|
_ => Err(ParsingError::MissingChild),
|
2024-03-06 19:58:41 +00:00
|
|
|
}
|
2024-03-06 17:35:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 19:58:41 +00:00
|
|
|
impl QRead<Status> for Status {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "status").await?;
|
2024-03-06 19:58:41 +00:00
|
|
|
let fullcode = xml.tag_string().await?;
|
|
|
|
let txtcode = fullcode.splitn(3, ' ').nth(1).ok_or(ParsingError::InvalidValue)?;
|
|
|
|
let code = http::status::StatusCode::from_bytes(txtcode.as_bytes()).or(Err(ParsingError::InvalidValue))?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Status(code))
|
2024-03-06 19:58:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 17:35:54 +00:00
|
|
|
impl QRead<ResponseDescription> for ResponseDescription {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "responsedescription").await?;
|
2024-03-06 17:35:54 +00:00
|
|
|
let cnt = xml.tag_string().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(ResponseDescription(cnt))
|
2024-03-06 17:35:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 19:58:41 +00:00
|
|
|
impl QRead<Location> for Location {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "location").await?;
|
|
|
|
let href = xml.find::<Href>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Location(href))
|
2024-03-06 19:58:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 11:42:27 +00:00
|
|
|
impl<E: Extension> QRead<PropertyUpdateItem<E>> for PropertyUpdateItem<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
match Remove::qread(xml).await {
|
|
|
|
Err(ParsingError::Recoverable) => (),
|
|
|
|
otherwise => return otherwise.map(PropertyUpdateItem::Remove),
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
2024-03-06 22:24:54 +00:00
|
|
|
Set::qread(xml).await.map(PropertyUpdateItem::Set)
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<E: Extension> QRead<Remove<E>> for Remove<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "remove").await?;
|
|
|
|
let propname = xml.find::<PropName<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Remove(propname))
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<E: Extension> QRead<Set<E>> for Set<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "set").await?;
|
|
|
|
let propvalue = xml.find::<PropValue<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Set(propvalue))
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-06 09:12:02 +00:00
|
|
|
impl<E: Extension> QRead<Violation<E>> for Violation<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
2024-03-07 11:25:22 +00:00
|
|
|
if xml.maybe_open(DAV_URN, "lock-token-matches-request-uri").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(Violation::LockTokenMatchesRequestUri)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "lock-token-submitted").await?.is_some() {
|
|
|
|
let links = xml.collect::<Href>().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(Violation::LockTokenSubmitted(links))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "no-conflicting-lock").await?.is_some() {
|
|
|
|
let links = xml.collect::<Href>().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(Violation::NoConflictingLock(links))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "no-external-entities").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(Violation::NoExternalEntities)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "preserved-live-properties").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(Violation::PreservedLiveProperties)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "propfind-finite-depth").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(Violation::PropfindFiniteDepth)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "cannot-modify-protected-property").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(Violation::CannotModifyProtectedProperty)
|
|
|
|
} else {
|
|
|
|
E::Error::qread(xml).await.map(Violation::Extension)
|
2024-03-06 09:12:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-03-04 21:27:37 +00:00
|
|
|
|
2024-03-05 18:06:04 +00:00
|
|
|
impl<E: Extension> QRead<Include<E>> for Include<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "include").await?;
|
|
|
|
let acc = xml.collect::<PropertyRequest<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Include(acc))
|
2024-03-04 21:27:37 +00:00
|
|
|
}
|
|
|
|
}
|
2024-03-04 16:55:48 +00:00
|
|
|
|
2024-03-05 18:06:04 +00:00
|
|
|
impl<E: Extension> QRead<PropName<E>> for PropName<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "prop").await?;
|
|
|
|
let acc = xml.collect::<PropertyRequest<E>>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(PropName(acc))
|
2024-03-04 21:27:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-05 18:06:04 +00:00
|
|
|
impl<E: Extension> QRead<PropertyRequest<E>> for PropertyRequest<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
2024-03-07 11:25:22 +00:00
|
|
|
let maybe = if xml.maybe_open(DAV_URN, "creationdate").await?.is_some() {
|
|
|
|
Some(PropertyRequest::CreationDate)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "displayname").await?.is_some() {
|
|
|
|
Some(PropertyRequest::DisplayName)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getcontentlanguage").await?.is_some() {
|
|
|
|
Some(PropertyRequest::GetContentLanguage)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getcontentlength").await?.is_some() {
|
|
|
|
Some(PropertyRequest::GetContentLength)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getcontenttype").await?.is_some() {
|
|
|
|
Some(PropertyRequest::GetContentType)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getetag").await?.is_some() {
|
|
|
|
Some(PropertyRequest::GetEtag)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getlastmodified").await?.is_some() {
|
|
|
|
Some(PropertyRequest::GetLastModified)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "lockdiscovery").await?.is_some() {
|
|
|
|
Some(PropertyRequest::LockDiscovery)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "resourcetype").await?.is_some() {
|
|
|
|
Some(PropertyRequest::ResourceType)
|
|
|
|
} else if xml.maybe_open(DAV_URN, "supportedlock").await?.is_some() {
|
|
|
|
Some(PropertyRequest::SupportedLock)
|
|
|
|
} else {
|
|
|
|
None
|
2024-03-06 11:42:27 +00:00
|
|
|
};
|
2024-03-05 15:07:47 +00:00
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
match maybe {
|
|
|
|
Some(pr) => {
|
|
|
|
xml.close().await?;
|
|
|
|
Ok(pr)
|
|
|
|
},
|
|
|
|
None => E::PropertyRequest::qread(xml).await.map(PropertyRequest::Extension),
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<E: Extension> QRead<Property<E>> for Property<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
2024-03-07 11:25:22 +00:00
|
|
|
// Core WebDAV properties
|
|
|
|
if xml.maybe_open(DAV_URN, "creationdate").await?.is_some() {
|
|
|
|
let datestr = xml.tag_string().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::CreationDate(DateTime::parse_from_rfc3339(datestr.as_str())?))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "displayname").await?.is_some() {
|
|
|
|
let name = xml.tag_string().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::DisplayName(name))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getcontentlanguage").await?.is_some() {
|
|
|
|
let lang = xml.tag_string().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::GetContentLanguage(lang))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getcontentlength").await?.is_some() {
|
|
|
|
let cl = xml.tag_string().await?.parse::<u64>()?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::GetContentLength(cl))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getcontenttype").await?.is_some() {
|
|
|
|
let ct = xml.tag_string().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::GetContentType(ct))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getetag").await?.is_some() {
|
|
|
|
let etag = xml.tag_string().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::GetEtag(etag))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "getlastmodified").await?.is_some() {
|
|
|
|
let datestr = xml.tag_string().await?;
|
|
|
|
xml.close().await?;
|
2024-03-07 13:25:08 +00:00
|
|
|
return Ok(Property::GetLastModified(DateTime::parse_from_rfc2822(datestr.as_str())?))
|
2024-03-07 11:25:22 +00:00
|
|
|
} else if xml.maybe_open(DAV_URN, "lockdiscovery").await?.is_some() {
|
|
|
|
let acc = xml.collect::<ActiveLock>().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::LockDiscovery(acc))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "resourcetype").await?.is_some() {
|
|
|
|
let acc = xml.collect::<ResourceType<E>>().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::ResourceType(acc))
|
|
|
|
} else if xml.maybe_open(DAV_URN, "supportedlock").await?.is_some() {
|
|
|
|
let acc = xml.collect::<LockEntry>().await?;
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(Property::SupportedLock(acc))
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Option 2: an extension property, delegating
|
2024-03-06 22:24:54 +00:00
|
|
|
E::Property::qread(xml).await.map(Property::Extension)
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<ActiveLock> for ActiveLock {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "activelock").await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
let (mut m_scope, mut m_type, mut m_depth, mut owner, mut timeout, mut locktoken, mut m_root) =
|
|
|
|
(None, None, None, None, None, None, None);
|
|
|
|
|
|
|
|
loop {
|
2024-03-06 22:24:54 +00:00
|
|
|
let mut dirty = false;
|
|
|
|
xml.maybe_read::<LockScope>(&mut m_scope, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<LockType>(&mut m_type, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Depth>(&mut m_depth, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Owner>(&mut owner, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<Timeout>(&mut timeout, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<LockToken>(&mut locktoken, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<LockRoot>(&mut m_root, &mut dirty).await?;
|
|
|
|
|
|
|
|
if !dirty {
|
2024-03-06 15:09:20 +00:00
|
|
|
match xml.peek() {
|
|
|
|
Event::End(_) => break,
|
|
|
|
_ => { xml.skip().await?; },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
match (m_scope, m_type, m_depth, m_root) {
|
|
|
|
(Some(lockscope), Some(locktype), Some(depth), Some(lockroot)) =>
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(ActiveLock { lockscope, locktype, depth, owner, timeout, locktoken, lockroot }),
|
2024-03-06 15:09:20 +00:00
|
|
|
_ => Err(ParsingError::MissingChild),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<Depth> for Depth {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "depth").await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
let depth_str = xml.tag_string().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
match depth_str.as_str() {
|
2024-03-06 22:24:54 +00:00
|
|
|
"0" => Ok(Depth::Zero),
|
|
|
|
"1" => Ok(Depth::One),
|
|
|
|
"infinity" => Ok(Depth::Infinity),
|
2024-03-06 15:09:20 +00:00
|
|
|
_ => Err(ParsingError::WrongToken),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<Owner> for Owner {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "owner").await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
|
|
|
|
let mut owner = Owner::Unknown;
|
|
|
|
loop {
|
|
|
|
match xml.peek() {
|
|
|
|
Event::Text(_) | Event::CData(_) => {
|
|
|
|
let txt = xml.tag_string().await?;
|
|
|
|
if matches!(owner, Owner::Unknown) {
|
|
|
|
owner = Owner::Txt(txt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Event::Start(_) | Event::Empty(_) => {
|
2024-03-06 22:24:54 +00:00
|
|
|
match Href::qread(xml).await {
|
|
|
|
Ok(href) => { owner = Owner::Href(href); },
|
|
|
|
Err(ParsingError::Recoverable) => { xml.skip().await?; },
|
|
|
|
Err(e) => return Err(e),
|
2024-03-06 15:09:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
Event::End(_) => break,
|
|
|
|
_ => { xml.skip().await?; },
|
|
|
|
}
|
|
|
|
};
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(owner)
|
2024-03-06 15:09:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<Timeout> for Timeout {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
2024-03-08 10:42:44 +00:00
|
|
|
const SEC_PFX: &str = "Second-";
|
2024-03-06 22:24:54 +00:00
|
|
|
xml.open(DAV_URN, "timeout").await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
|
|
|
|
let timeout = match xml.tag_string().await?.as_str() {
|
|
|
|
"Infinite" => Timeout::Infinite,
|
|
|
|
seconds => match seconds.strip_prefix(SEC_PFX) {
|
|
|
|
Some(secs) => Timeout::Seconds(secs.parse::<u32>()?),
|
|
|
|
None => return Err(ParsingError::InvalidValue),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(timeout)
|
2024-03-06 15:09:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<LockToken> for LockToken {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "locktoken").await?;
|
2024-03-08 10:42:44 +00:00
|
|
|
let href = xml.find::<Href>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(LockToken(href))
|
2024-03-06 15:09:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<LockRoot> for LockRoot {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "lockroot").await?;
|
2024-03-08 10:42:44 +00:00
|
|
|
let href = xml.find::<Href>().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(LockRoot(href))
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<E: Extension> QRead<ResourceType<E>> for ResourceType<E> {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
2024-03-07 11:25:22 +00:00
|
|
|
if xml.maybe_open(DAV_URN, "collection").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
return Ok(ResourceType::Collection)
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
2024-03-07 11:25:22 +00:00
|
|
|
|
|
|
|
E::ResourceType::qread(xml).await.map(ResourceType::Extension)
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<LockEntry> for LockEntry {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "lockentry").await?;
|
2024-03-06 11:42:27 +00:00
|
|
|
let (mut maybe_scope, mut maybe_type) = (None, None);
|
|
|
|
|
2024-03-06 09:12:02 +00:00
|
|
|
loop {
|
2024-03-06 22:24:54 +00:00
|
|
|
let mut dirty = false;
|
|
|
|
xml.maybe_read::<LockScope>(&mut maybe_scope, &mut dirty).await?;
|
|
|
|
xml.maybe_read::<LockType>(&mut maybe_type, &mut dirty).await?;
|
|
|
|
if !dirty {
|
|
|
|
match xml.peek() {
|
|
|
|
Event::End(_) => break,
|
|
|
|
_ => xml.skip().await?,
|
|
|
|
};
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
match (maybe_scope, maybe_type) {
|
|
|
|
(Some(lockscope), Some(locktype)) => Ok(LockEntry { lockscope, locktype }),
|
|
|
|
_ => Err(ParsingError::MissingChild),
|
|
|
|
}
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<LockScope> for LockScope {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "lockscope").await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
|
2024-03-06 11:42:27 +00:00
|
|
|
let lockscope = loop {
|
2024-03-07 11:25:22 +00:00
|
|
|
if xml.maybe_open(DAV_URN, "exclusive").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
break LockScope::Exclusive
|
2024-03-08 17:23:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if xml.maybe_open(DAV_URN, "shared").await?.is_some() {
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
|
|
|
break LockScope::Shared
|
|
|
|
}
|
|
|
|
|
|
|
|
xml.skip().await?;
|
2024-03-06 11:42:27 +00:00
|
|
|
};
|
|
|
|
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(lockscope)
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<LockType> for LockType {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "locktype").await?;
|
2024-03-06 15:09:20 +00:00
|
|
|
|
2024-03-06 11:42:27 +00:00
|
|
|
let locktype = loop {
|
2024-03-07 11:25:22 +00:00
|
|
|
if xml.maybe_open(DAV_URN, "write").await?.is_some() {
|
|
|
|
xml.close().await?;
|
|
|
|
break LockType::Write
|
|
|
|
}
|
|
|
|
|
|
|
|
xml.skip().await?;
|
2024-03-06 11:42:27 +00:00
|
|
|
};
|
2024-03-07 11:25:22 +00:00
|
|
|
|
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(locktype)
|
2024-03-06 11:42:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl QRead<Href> for Href {
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn qread(xml: &mut Reader<impl IRead>) -> Result<Self, ParsingError> {
|
|
|
|
xml.open(DAV_URN, "href").await?;
|
2024-03-08 08:55:33 +00:00
|
|
|
let url = xml.tag_string().await?;
|
2024-03-07 11:25:22 +00:00
|
|
|
xml.close().await?;
|
2024-03-06 22:24:54 +00:00
|
|
|
Ok(Href(url))
|
2024-03-06 09:12:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-04 16:55:48 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
2024-03-08 10:34:24 +00:00
|
|
|
use chrono::{FixedOffset, TimeZone};
|
2024-03-08 07:17:03 +00:00
|
|
|
use crate::realization::Core;
|
2024-03-08 10:34:24 +00:00
|
|
|
use quick_xml::reader::NsReader;
|
2024-03-04 12:36:41 +00:00
|
|
|
|
2024-03-04 16:55:48 +00:00
|
|
|
#[tokio::test]
|
2024-03-05 15:07:47 +00:00
|
|
|
async fn basic_propfind_propname() {
|
2024-03-04 21:27:37 +00:00
|
|
|
let src = r#"<?xml version="1.0" encoding="utf-8" ?>
|
|
|
|
<rando/>
|
|
|
|
<garbage><old/></garbage>
|
|
|
|
<D:propfind xmlns:D="DAV:">
|
|
|
|
<D:propname/>
|
|
|
|
</D:propfind>
|
|
|
|
"#;
|
2024-03-04 12:36:41 +00:00
|
|
|
|
2024-03-05 18:06:04 +00:00
|
|
|
let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
|
2024-03-06 22:24:54 +00:00
|
|
|
let got = rdr.find::<PropFind::<Core>>().await.unwrap();
|
2024-03-05 18:06:04 +00:00
|
|
|
|
|
|
|
assert_eq!(got, PropFind::<Core>::PropName);
|
2024-03-04 12:36:41 +00:00
|
|
|
}
|
2024-03-05 18:06:04 +00:00
|
|
|
|
2024-03-05 15:07:47 +00:00
|
|
|
#[tokio::test]
|
|
|
|
async fn basic_propfind_prop() {
|
|
|
|
let src = r#"<?xml version="1.0" encoding="utf-8" ?>
|
|
|
|
<rando/>
|
|
|
|
<garbage><old/></garbage>
|
|
|
|
<D:propfind xmlns:D="DAV:">
|
|
|
|
<D:prop>
|
2024-03-05 18:06:04 +00:00
|
|
|
<D:displayname/>
|
|
|
|
<D:getcontentlength/>
|
|
|
|
<D:getcontenttype/>
|
|
|
|
<D:getetag/>
|
|
|
|
<D:getlastmodified/>
|
|
|
|
<D:resourcetype/>
|
|
|
|
<D:supportedlock/>
|
2024-03-05 15:07:47 +00:00
|
|
|
</D:prop>
|
|
|
|
</D:propfind>
|
|
|
|
"#;
|
|
|
|
|
2024-03-05 18:06:04 +00:00
|
|
|
let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
|
2024-03-06 22:24:54 +00:00
|
|
|
let got = rdr.find::<PropFind::<Core>>().await.unwrap();
|
2024-03-05 18:06:04 +00:00
|
|
|
|
2024-03-05 15:07:47 +00:00
|
|
|
assert_eq!(got, PropFind::Prop(PropName(vec![
|
|
|
|
PropertyRequest::DisplayName,
|
|
|
|
PropertyRequest::GetContentLength,
|
|
|
|
PropertyRequest::GetContentType,
|
|
|
|
PropertyRequest::GetEtag,
|
|
|
|
PropertyRequest::GetLastModified,
|
|
|
|
PropertyRequest::ResourceType,
|
|
|
|
PropertyRequest::SupportedLock,
|
|
|
|
])));
|
|
|
|
}
|
2024-03-06 09:12:02 +00:00
|
|
|
|
|
|
|
#[tokio::test]
|
|
|
|
async fn rfc_lock_error() {
|
|
|
|
let src = r#"<?xml version="1.0" encoding="utf-8" ?>
|
|
|
|
<D:error xmlns:D="DAV:">
|
|
|
|
<D:lock-token-submitted>
|
|
|
|
<D:href>/locked/</D:href>
|
|
|
|
</D:lock-token-submitted>
|
|
|
|
</D:error>"#;
|
|
|
|
|
|
|
|
let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
|
2024-03-06 22:24:54 +00:00
|
|
|
let got = rdr.find::<Error::<Core>>().await.unwrap();
|
2024-03-06 09:12:02 +00:00
|
|
|
|
|
|
|
assert_eq!(got, Error(vec![
|
|
|
|
Violation::LockTokenSubmitted(vec![
|
|
|
|
Href("/locked/".into())
|
|
|
|
])
|
|
|
|
]));
|
|
|
|
}
|
2024-03-06 15:09:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
#[tokio::test]
|
|
|
|
async fn rfc_propertyupdate() {
|
|
|
|
let src = r#"<?xml version="1.0" encoding="utf-8" ?>
|
|
|
|
<D:propertyupdate xmlns:D="DAV:"
|
|
|
|
xmlns:Z="http://ns.example.com/standards/z39.50/">
|
|
|
|
<D:set>
|
|
|
|
<D:prop>
|
|
|
|
<Z:Authors>
|
|
|
|
<Z:Author>Jim Whitehead</Z:Author>
|
|
|
|
<Z:Author>Roy Fielding</Z:Author>
|
|
|
|
</Z:Authors>
|
|
|
|
</D:prop>
|
|
|
|
</D:set>
|
|
|
|
<D:remove>
|
|
|
|
<D:prop><Z:Copyright-Owner/></D:prop>
|
|
|
|
</D:remove>
|
|
|
|
</D:propertyupdate>"#;
|
|
|
|
|
|
|
|
let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
|
2024-03-06 22:24:54 +00:00
|
|
|
let got = rdr.find::<PropertyUpdate::<Core>>().await.unwrap();
|
2024-03-06 15:09:20 +00:00
|
|
|
|
|
|
|
assert_eq!(got, PropertyUpdate(vec![
|
|
|
|
PropertyUpdateItem::Set(Set(PropValue(vec![]))),
|
|
|
|
PropertyUpdateItem::Remove(Remove(PropName(vec![]))),
|
|
|
|
]));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::test]
|
2024-03-06 22:24:54 +00:00
|
|
|
async fn rfc_lockinfo() {
|
2024-03-06 15:09:20 +00:00
|
|
|
let src = r#"
|
|
|
|
<?xml version="1.0" encoding="utf-8" ?>
|
|
|
|
<D:lockinfo xmlns:D='DAV:'>
|
|
|
|
<D:lockscope><D:exclusive/></D:lockscope>
|
|
|
|
<D:locktype><D:write/></D:locktype>
|
|
|
|
<D:owner>
|
|
|
|
<D:href>http://example.org/~ejw/contact.html</D:href>
|
|
|
|
</D:owner>
|
|
|
|
</D:lockinfo>
|
|
|
|
"#;
|
|
|
|
|
|
|
|
let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
|
2024-03-06 22:24:54 +00:00
|
|
|
let got = rdr.find::<LockInfo>().await.unwrap();
|
|
|
|
|
2024-03-06 15:09:20 +00:00
|
|
|
assert_eq!(got, LockInfo {
|
|
|
|
lockscope: LockScope::Exclusive,
|
|
|
|
locktype: LockType::Write,
|
|
|
|
owner: Some(Owner::Href(Href("http://example.org/~ejw/contact.html".into()))),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2024-03-06 22:24:54 +00:00
|
|
|
#[tokio::test]
|
|
|
|
async fn rfc_multistatus_name() {
|
|
|
|
let src = r#"
|
|
|
|
<?xml version="1.0" encoding="utf-8" ?>
|
|
|
|
<multistatus xmlns="DAV:">
|
|
|
|
<response>
|
|
|
|
<href>http://www.example.com/container/</href>
|
|
|
|
<propstat>
|
|
|
|
<prop xmlns:R="http://ns.example.com/boxschema/">
|
|
|
|
<R:bigbox/>
|
|
|
|
<R:author/>
|
|
|
|
<creationdate/>
|
|
|
|
<displayname/>
|
|
|
|
<resourcetype/>
|
|
|
|
<supportedlock/>
|
|
|
|
</prop>
|
|
|
|
<status>HTTP/1.1 200 OK</status>
|
|
|
|
</propstat>
|
|
|
|
</response>
|
|
|
|
<response>
|
|
|
|
<href>http://www.example.com/container/front.html</href>
|
|
|
|
<propstat>
|
|
|
|
<prop xmlns:R="http://ns.example.com/boxschema/">
|
|
|
|
<R:bigbox/>
|
|
|
|
<creationdate/>
|
|
|
|
<displayname/>
|
|
|
|
<getcontentlength/>
|
|
|
|
<getcontenttype/>
|
|
|
|
<getetag/>
|
|
|
|
<getlastmodified/>
|
|
|
|
<resourcetype/>
|
|
|
|
<supportedlock/>
|
|
|
|
</prop>
|
|
|
|
<status>HTTP/1.1 200 OK</status>
|
|
|
|
</propstat>
|
|
|
|
</response>
|
|
|
|
</multistatus>
|
|
|
|
"#;
|
|
|
|
|
|
|
|
let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
|
|
|
|
let got = rdr.find::<Multistatus::<Core, PropName<Core>>>().await.unwrap();
|
|
|
|
|
2024-03-07 08:49:09 +00:00
|
|
|
assert_eq!(got, Multistatus {
|
2024-03-06 22:24:54 +00:00
|
|
|
responses: vec![
|
|
|
|
Response {
|
2024-03-07 08:49:09 +00:00
|
|
|
status_or_propstat: StatusOrPropstat::PropStat(
|
|
|
|
Href("http://www.example.com/container/".into()),
|
|
|
|
vec![PropStat {
|
|
|
|
prop: PropName(vec![
|
|
|
|
PropertyRequest::CreationDate,
|
|
|
|
PropertyRequest::DisplayName,
|
|
|
|
PropertyRequest::ResourceType,
|
|
|
|
PropertyRequest::SupportedLock,
|
|
|
|
]),
|
|
|
|
status: Status(http::status::StatusCode::OK),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
}],
|
|
|
|
),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
location: None,
|
|
|
|
},
|
|
|
|
Response {
|
|
|
|
status_or_propstat: StatusOrPropstat::PropStat(
|
|
|
|
Href("http://www.example.com/container/front.html".into()),
|
|
|
|
vec![PropStat {
|
|
|
|
prop: PropName(vec![
|
|
|
|
PropertyRequest::CreationDate,
|
|
|
|
PropertyRequest::DisplayName,
|
|
|
|
PropertyRequest::GetContentLength,
|
|
|
|
PropertyRequest::GetContentType,
|
|
|
|
PropertyRequest::GetEtag,
|
|
|
|
PropertyRequest::GetLastModified,
|
|
|
|
PropertyRequest::ResourceType,
|
|
|
|
PropertyRequest::SupportedLock,
|
|
|
|
]),
|
|
|
|
status: Status(http::status::StatusCode::OK),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
}],
|
|
|
|
),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
location: None,
|
2024-03-06 22:24:54 +00:00
|
|
|
},
|
|
|
|
],
|
|
|
|
responsedescription: None,
|
2024-03-07 08:49:09 +00:00
|
|
|
});
|
2024-03-06 22:24:54 +00:00
|
|
|
}
|
2024-03-07 11:25:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
#[tokio::test]
|
|
|
|
async fn rfc_multistatus_value() {
|
|
|
|
let src = r#"
|
|
|
|
<?xml version="1.0" encoding="utf-8" ?>
|
|
|
|
<D:multistatus xmlns:D="DAV:">
|
|
|
|
<D:response>
|
|
|
|
<D:href>/container/</D:href>
|
|
|
|
<D:propstat>
|
|
|
|
<D:prop xmlns:R="http://ns.example.com/boxschema/">
|
|
|
|
<R:bigbox><R:BoxType>Box type A</R:BoxType></R:bigbox>
|
|
|
|
<R:author><R:Name>Hadrian</R:Name></R:author>
|
|
|
|
<D:creationdate>1997-12-01T17:42:21-08:00</D:creationdate>
|
|
|
|
<D:displayname>Example collection</D:displayname>
|
|
|
|
<D:resourcetype><D:collection/></D:resourcetype>
|
|
|
|
<D:supportedlock>
|
|
|
|
<D:lockentry>
|
|
|
|
<D:lockscope><D:exclusive/></D:lockscope>
|
|
|
|
<D:locktype><D:write/></D:locktype>
|
|
|
|
</D:lockentry>
|
|
|
|
<D:lockentry>
|
|
|
|
<D:lockscope><D:shared/></D:lockscope>
|
|
|
|
<D:locktype><D:write/></D:locktype>
|
|
|
|
</D:lockentry>
|
|
|
|
</D:supportedlock>
|
|
|
|
</D:prop>
|
|
|
|
<D:status>HTTP/1.1 200 OK</D:status>
|
|
|
|
</D:propstat>
|
|
|
|
</D:response>
|
|
|
|
<D:response>
|
|
|
|
<D:href>/container/front.html</D:href>
|
|
|
|
<D:propstat>
|
|
|
|
<D:prop xmlns:R="http://ns.example.com/boxschema/">
|
|
|
|
<R:bigbox><R:BoxType>Box type B</R:BoxType>
|
|
|
|
</R:bigbox>
|
|
|
|
<D:creationdate>1997-12-01T18:27:21-08:00</D:creationdate>
|
|
|
|
<D:displayname>Example HTML resource</D:displayname>
|
|
|
|
<D:getcontentlength>4525</D:getcontentlength>
|
|
|
|
<D:getcontenttype>text/html</D:getcontenttype>
|
|
|
|
<D:getetag>"zzyzx"</D:getetag>
|
|
|
|
<D:getlastmodified
|
|
|
|
>Mon, 12 Jan 1998 09:25:56 GMT</D:getlastmodified>
|
|
|
|
<D:resourcetype/>
|
|
|
|
<D:supportedlock>
|
|
|
|
<D:lockentry>
|
|
|
|
<D:lockscope><D:exclusive/></D:lockscope>
|
|
|
|
<D:locktype><D:write/></D:locktype>
|
|
|
|
</D:lockentry>
|
|
|
|
<D:lockentry>
|
|
|
|
<D:lockscope><D:shared/></D:lockscope>
|
|
|
|
<D:locktype><D:write/></D:locktype>
|
|
|
|
</D:lockentry>
|
|
|
|
</D:supportedlock>
|
|
|
|
</D:prop>
|
|
|
|
<D:status>HTTP/1.1 200 OK</D:status>
|
|
|
|
</D:propstat>
|
|
|
|
</D:response>
|
|
|
|
</D:multistatus>"#;
|
|
|
|
|
|
|
|
let mut rdr = Reader::new(NsReader::from_reader(src.as_bytes())).await.unwrap();
|
|
|
|
let got = rdr.find::<Multistatus::<Core, PropValue<Core>>>().await.unwrap();
|
2024-03-07 13:25:08 +00:00
|
|
|
|
|
|
|
assert_eq!(got, Multistatus {
|
|
|
|
responses: vec![
|
|
|
|
Response {
|
|
|
|
status_or_propstat: StatusOrPropstat::PropStat(
|
|
|
|
Href("/container/".into()),
|
|
|
|
vec![PropStat {
|
|
|
|
prop: PropValue(vec![
|
|
|
|
Property::CreationDate(FixedOffset::west_opt(8 * 3600).unwrap().with_ymd_and_hms(1997, 12, 01, 17, 42, 21).unwrap()),
|
|
|
|
Property::DisplayName("Example collection".into()),
|
|
|
|
Property::ResourceType(vec![ResourceType::Collection]),
|
|
|
|
Property::SupportedLock(vec![
|
|
|
|
LockEntry {
|
|
|
|
lockscope: LockScope::Exclusive,
|
|
|
|
locktype: LockType::Write,
|
|
|
|
},
|
|
|
|
LockEntry {
|
|
|
|
lockscope: LockScope::Shared,
|
|
|
|
locktype: LockType::Write,
|
|
|
|
},
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
status: Status(http::status::StatusCode::OK),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
}],
|
|
|
|
),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
location: None,
|
|
|
|
|
|
|
|
},
|
|
|
|
Response {
|
|
|
|
status_or_propstat: StatusOrPropstat::PropStat(
|
|
|
|
Href("/container/front.html".into()),
|
|
|
|
vec![PropStat {
|
|
|
|
prop: PropValue(vec![
|
|
|
|
Property::CreationDate(FixedOffset::west_opt(8 * 3600).unwrap().with_ymd_and_hms(1997, 12, 01, 18, 27, 21).unwrap()),
|
|
|
|
Property::DisplayName("Example HTML resource".into()),
|
|
|
|
Property::GetContentLength(4525),
|
|
|
|
Property::GetContentType("text/html".into()),
|
|
|
|
Property::GetEtag(r#""zzyzx""#.into()),
|
|
|
|
Property::GetLastModified(FixedOffset::west_opt(0).unwrap().with_ymd_and_hms(1998, 01, 12, 09, 25, 56).unwrap()),
|
2024-03-08 10:34:24 +00:00
|
|
|
Property::ResourceType(vec![]),
|
2024-03-07 13:25:08 +00:00
|
|
|
Property::SupportedLock(vec![
|
|
|
|
LockEntry {
|
|
|
|
lockscope: LockScope::Exclusive,
|
|
|
|
locktype: LockType::Write,
|
|
|
|
},
|
|
|
|
LockEntry {
|
|
|
|
lockscope: LockScope::Shared,
|
|
|
|
locktype: LockType::Write,
|
|
|
|
},
|
|
|
|
]),
|
|
|
|
]),
|
|
|
|
status: Status(http::status::StatusCode::OK),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
}],
|
|
|
|
),
|
|
|
|
error: None,
|
|
|
|
responsedescription: None,
|
|
|
|
location: None,
|
|
|
|
|
|
|
|
},
|
|
|
|
],
|
|
|
|
responsedescription: None,
|
|
|
|
});
|
2024-03-07 11:25:22 +00:00
|
|
|
}
|
|
|
|
|
2024-03-04 12:36:41 +00:00
|
|
|
}
|