mirror of
https://github.com/lennart-k/rustical.git
synced 2025-12-13 21:42:34 +00:00
@@ -33,7 +33,7 @@ pub async fn get_objects_calendar_multiget<C: CalendarStore>(
|
|||||||
cal_id: &str,
|
cal_id: &str,
|
||||||
store: &C,
|
store: &C,
|
||||||
) -> Result<(Vec<CalendarObject>, Vec<String>), Error> {
|
) -> Result<(Vec<CalendarObject>, Vec<String>), Error> {
|
||||||
let resource_def = ResourceDef::prefix(path).join(&ResourceDef::new("/{object_id}"));
|
let resource_def = ResourceDef::prefix(path).join(&ResourceDef::new("/{object_id}.ics"));
|
||||||
|
|
||||||
let mut result = vec![];
|
let mut result = vec![];
|
||||||
let mut not_found = vec![];
|
let mut not_found = vec![];
|
||||||
@@ -42,6 +42,7 @@ pub async fn get_objects_calendar_multiget<C: CalendarStore>(
|
|||||||
let mut path = Path::new(href.as_str());
|
let mut path = Path::new(href.as_str());
|
||||||
if !resource_def.capture_match_info(&mut path) {
|
if !resource_def.capture_match_info(&mut path) {
|
||||||
not_found.push(href.to_owned());
|
not_found.push(href.to_owned());
|
||||||
|
continue;
|
||||||
};
|
};
|
||||||
let object_id = path.get("object_id").unwrap();
|
let object_id = path.get("object_id").unwrap();
|
||||||
match store.get_object(principal, cal_id, object_id).await {
|
match store.get_object(principal, cal_id, object_id).await {
|
||||||
@@ -85,7 +86,7 @@ pub async fn handle_calendar_multiget<C: CalendarStore>(
|
|||||||
|
|
||||||
let mut responses = Vec::new();
|
let mut responses = Vec::new();
|
||||||
for object in objects {
|
for object in objects {
|
||||||
let path = format!("{}/{}", req.path(), object.get_id());
|
let path = format!("{}/{}.ics", req.path(), object.get_id());
|
||||||
responses.push(
|
responses.push(
|
||||||
CalendarObjectResource {
|
CalendarObjectResource {
|
||||||
object,
|
object,
|
||||||
|
|||||||
@@ -244,7 +244,11 @@ pub async fn handle_calendar_query<C: CalendarStore>(
|
|||||||
|
|
||||||
let mut responses = Vec::new();
|
let mut responses = Vec::new();
|
||||||
for object in objects {
|
for object in objects {
|
||||||
let path = format!("{}/{}", req.path().trim_end_matches('/'), object.get_id());
|
let path = format!(
|
||||||
|
"{}/{}.ics",
|
||||||
|
req.path().trim_end_matches('/'),
|
||||||
|
object.get_id()
|
||||||
|
);
|
||||||
responses.push(
|
responses.push(
|
||||||
CalendarObjectResource {
|
CalendarObjectResource {
|
||||||
object,
|
object,
|
||||||
|
|||||||
@@ -49,7 +49,11 @@ pub async fn handle_sync_collection<C: CalendarStore>(
|
|||||||
|
|
||||||
let mut responses = Vec::new();
|
let mut responses = Vec::new();
|
||||||
for object in new_objects {
|
for object in new_objects {
|
||||||
let path = format!("{}/{}", req.path().trim_end_matches('/'), object.get_id());
|
let path = format!(
|
||||||
|
"{}/{}.ics",
|
||||||
|
req.path().trim_end_matches('/'),
|
||||||
|
object.get_id()
|
||||||
|
);
|
||||||
responses.push(
|
responses.push(
|
||||||
CalendarObjectResource {
|
CalendarObjectResource {
|
||||||
object,
|
object,
|
||||||
@@ -60,7 +64,7 @@ pub async fn handle_sync_collection<C: CalendarStore>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
for object_id in deleted_objects {
|
for object_id in deleted_objects {
|
||||||
let path = format!("{}/{}", req.path().trim_end_matches('/'), object_id);
|
let path = format!("{}/{}.ics", req.path().trim_end_matches('/'), object_id);
|
||||||
responses.push(ResponseElement {
|
responses.push(ResponseElement {
|
||||||
href: path,
|
href: path,
|
||||||
status: Some(StatusCode::NOT_FOUND),
|
status: Some(StatusCode::NOT_FOUND),
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
use crate::Error;
|
use crate::Error;
|
||||||
|
use actix_web::HttpRequest;
|
||||||
|
use actix_web::HttpResponse;
|
||||||
use actix_web::http::header;
|
use actix_web::http::header;
|
||||||
use actix_web::http::header::HeaderValue;
|
use actix_web::http::header::HeaderValue;
|
||||||
use actix_web::web::{Data, Path};
|
use actix_web::web::{Data, Path};
|
||||||
use actix_web::HttpRequest;
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use rustical_store::auth::User;
|
use rustical_store::auth::User;
|
||||||
use rustical_store::{CalendarObject, CalendarStore};
|
use rustical_store::{CalendarObject, CalendarStore};
|
||||||
use tracing::instrument;
|
use tracing::instrument;
|
||||||
@@ -20,7 +20,7 @@ pub async fn get_event<C: CalendarStore>(
|
|||||||
) -> Result<HttpResponse, Error> {
|
) -> Result<HttpResponse, Error> {
|
||||||
let CalendarObjectPathComponents {
|
let CalendarObjectPathComponents {
|
||||||
principal,
|
principal,
|
||||||
cal_id,
|
calendar_id,
|
||||||
object_id,
|
object_id,
|
||||||
} = path.into_inner();
|
} = path.into_inner();
|
||||||
|
|
||||||
@@ -28,12 +28,14 @@ pub async fn get_event<C: CalendarStore>(
|
|||||||
return Ok(HttpResponse::Unauthorized().body(""));
|
return Ok(HttpResponse::Unauthorized().body(""));
|
||||||
}
|
}
|
||||||
|
|
||||||
let calendar = store.get_calendar(&principal, &cal_id).await?;
|
let calendar = store.get_calendar(&principal, &calendar_id).await?;
|
||||||
if !user.is_principal(&calendar.principal) {
|
if !user.is_principal(&calendar.principal) {
|
||||||
return Ok(HttpResponse::Unauthorized().body(""));
|
return Ok(HttpResponse::Unauthorized().body(""));
|
||||||
}
|
}
|
||||||
|
|
||||||
let event = store.get_object(&principal, &cal_id, &object_id).await?;
|
let event = store
|
||||||
|
.get_object(&principal, &calendar_id, &object_id)
|
||||||
|
.await?;
|
||||||
|
|
||||||
Ok(HttpResponse::Ok()
|
Ok(HttpResponse::Ok()
|
||||||
.insert_header(("ETag", event.get_etag()))
|
.insert_header(("ETag", event.get_etag()))
|
||||||
@@ -52,7 +54,7 @@ pub async fn put_event<C: CalendarStore>(
|
|||||||
) -> Result<HttpResponse, Error> {
|
) -> Result<HttpResponse, Error> {
|
||||||
let CalendarObjectPathComponents {
|
let CalendarObjectPathComponents {
|
||||||
principal,
|
principal,
|
||||||
cal_id,
|
calendar_id,
|
||||||
object_id,
|
object_id,
|
||||||
} = path.into_inner();
|
} = path.into_inner();
|
||||||
|
|
||||||
@@ -65,7 +67,7 @@ pub async fn put_event<C: CalendarStore>(
|
|||||||
|
|
||||||
let object = CalendarObject::from_ics(object_id, body)?;
|
let object = CalendarObject::from_ics(object_id, body)?;
|
||||||
store
|
store
|
||||||
.put_object(principal, cal_id, object, overwrite)
|
.put_object(principal, calendar_id, object, overwrite)
|
||||||
.await?;
|
.await?;
|
||||||
|
|
||||||
Ok(HttpResponse::Created().body(""))
|
Ok(HttpResponse::Created().body(""))
|
||||||
|
|||||||
@@ -105,31 +105,13 @@ impl Resource for CalendarObjectResource {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone, Deserialize)]
|
||||||
pub struct CalendarObjectPathComponents {
|
pub struct CalendarObjectPathComponents {
|
||||||
pub principal: String,
|
pub principal: String,
|
||||||
pub cal_id: String,
|
pub calendar_id: String,
|
||||||
pub object_id: String,
|
pub object_id: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'de> Deserialize<'de> for CalendarObjectPathComponents {
|
|
||||||
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
|
||||||
where
|
|
||||||
D: serde::Deserializer<'de>,
|
|
||||||
{
|
|
||||||
type Inner = (String, String, String);
|
|
||||||
let (principal, calendar, mut object) = Inner::deserialize(deserializer)?;
|
|
||||||
if object.ends_with(".ics") {
|
|
||||||
object.truncate(object.len() - 4);
|
|
||||||
}
|
|
||||||
Ok(Self {
|
|
||||||
principal,
|
|
||||||
cal_id: calendar,
|
|
||||||
object_id: object,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[async_trait(?Send)]
|
#[async_trait(?Send)]
|
||||||
impl<C: CalendarStore> ResourceService for CalendarObjectResourceService<C> {
|
impl<C: CalendarStore> ResourceService for CalendarObjectResourceService<C> {
|
||||||
type PathComponents = CalendarObjectPathComponents;
|
type PathComponents = CalendarObjectPathComponents;
|
||||||
@@ -142,13 +124,13 @@ impl<C: CalendarStore> ResourceService for CalendarObjectResourceService<C> {
|
|||||||
&self,
|
&self,
|
||||||
CalendarObjectPathComponents {
|
CalendarObjectPathComponents {
|
||||||
principal,
|
principal,
|
||||||
cal_id,
|
calendar_id,
|
||||||
object_id,
|
object_id,
|
||||||
}: &Self::PathComponents,
|
}: &Self::PathComponents,
|
||||||
) -> Result<Self::Resource, Self::Error> {
|
) -> Result<Self::Resource, Self::Error> {
|
||||||
let object = self
|
let object = self
|
||||||
.cal_store
|
.cal_store
|
||||||
.get_object(principal, cal_id, object_id)
|
.get_object(principal, calendar_id, object_id)
|
||||||
.await?;
|
.await?;
|
||||||
Ok(CalendarObjectResource {
|
Ok(CalendarObjectResource {
|
||||||
object,
|
object,
|
||||||
@@ -160,13 +142,13 @@ impl<C: CalendarStore> ResourceService for CalendarObjectResourceService<C> {
|
|||||||
&self,
|
&self,
|
||||||
CalendarObjectPathComponents {
|
CalendarObjectPathComponents {
|
||||||
principal,
|
principal,
|
||||||
cal_id,
|
calendar_id,
|
||||||
object_id,
|
object_id,
|
||||||
}: &Self::PathComponents,
|
}: &Self::PathComponents,
|
||||||
use_trashbin: bool,
|
use_trashbin: bool,
|
||||||
) -> Result<(), Self::Error> {
|
) -> Result<(), Self::Error> {
|
||||||
self.cal_store
|
self.cal_store
|
||||||
.delete_object(principal, cal_id, object_id, use_trashbin)
|
.delete_object(principal, calendar_id, object_id, use_trashbin)
|
||||||
.await?;
|
.await?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -75,22 +75,22 @@ pub fn caldav_service<
|
|||||||
.service(web::scope("/calendar")
|
.service(web::scope("/calendar")
|
||||||
.service(CalendarSetResourceService::new(store.clone()).actix_resource())
|
.service(CalendarSetResourceService::new(store.clone()).actix_resource())
|
||||||
.service(
|
.service(
|
||||||
web::scope("/{calendar}")
|
web::scope("/{calendar_id}")
|
||||||
.service(
|
.service(
|
||||||
ResourceServiceRoute(CalendarResourceService::<_, S>::new(store.clone()))
|
ResourceServiceRoute(CalendarResourceService::<_, S>::new(store.clone()))
|
||||||
)
|
)
|
||||||
.service(web::scope("/{object}").service(CalendarObjectResourceService::new(store.clone()).actix_resource()
|
.service(web::scope("/{object_id}.ics").service(CalendarObjectResourceService::new(store.clone()).actix_resource()
|
||||||
))
|
))
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
.service(web::scope("/birthdays")
|
.service(web::scope("/birthdays")
|
||||||
.service(CalendarSetResourceService::new(birthday_store.clone()).actix_resource())
|
.service(CalendarSetResourceService::new(birthday_store.clone()).actix_resource())
|
||||||
.service(
|
.service(
|
||||||
web::scope("/{calendar}")
|
web::scope("/{calendar_id}")
|
||||||
.service(
|
.service(
|
||||||
ResourceServiceRoute(CalendarResourceService::<_, S>::new(birthday_store.clone()))
|
ResourceServiceRoute(CalendarResourceService::<_, S>::new(birthday_store.clone()))
|
||||||
)
|
)
|
||||||
.service(web::scope("/{object}").service(CalendarObjectResourceService::new(birthday_store.clone()).actix_resource()
|
.service(web::scope("/{object_id}.ics").service(CalendarObjectResourceService::new(birthday_store.clone()).actix_resource()
|
||||||
))
|
))
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -101,31 +101,13 @@ impl Resource for AddressObjectResource {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone, Deserialize)]
|
||||||
pub struct AddressObjectPathComponents {
|
pub struct AddressObjectPathComponents {
|
||||||
pub principal: String,
|
pub principal: String,
|
||||||
pub addressbook_id: String,
|
pub addressbook_id: String,
|
||||||
pub object_id: String,
|
pub object_id: String,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'de> Deserialize<'de> for AddressObjectPathComponents {
|
|
||||||
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
|
||||||
where
|
|
||||||
D: serde::Deserializer<'de>,
|
|
||||||
{
|
|
||||||
type Inner = (String, String, String);
|
|
||||||
let (principal, addressbook_id, mut object) = Inner::deserialize(deserializer)?;
|
|
||||||
if object.ends_with(".vcf") {
|
|
||||||
object.truncate(object.len() - 4);
|
|
||||||
}
|
|
||||||
Ok(Self {
|
|
||||||
principal,
|
|
||||||
addressbook_id,
|
|
||||||
object_id: object,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[async_trait(?Send)]
|
#[async_trait(?Send)]
|
||||||
impl<AS: AddressbookStore> ResourceService for AddressObjectResourceService<AS> {
|
impl<AS: AddressbookStore> ResourceService for AddressObjectResourceService<AS> {
|
||||||
type PathComponents = AddressObjectPathComponents;
|
type PathComponents = AddressObjectPathComponents;
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ pub async fn get_objects_addressbook_multiget<AS: AddressbookStore>(
|
|||||||
addressbook_id: &str,
|
addressbook_id: &str,
|
||||||
store: &AS,
|
store: &AS,
|
||||||
) -> Result<(Vec<AddressObject>, Vec<String>), Error> {
|
) -> Result<(Vec<AddressObject>, Vec<String>), Error> {
|
||||||
let resource_def = ResourceDef::prefix(path).join(&ResourceDef::new("/{object_id}"));
|
let resource_def = ResourceDef::prefix(path).join(&ResourceDef::new("/{object_id}.vcf"));
|
||||||
|
|
||||||
let mut result = vec![];
|
let mut result = vec![];
|
||||||
let mut not_found = vec![];
|
let mut not_found = vec![];
|
||||||
@@ -83,7 +83,7 @@ pub async fn handle_addressbook_multiget<AS: AddressbookStore>(
|
|||||||
|
|
||||||
let mut responses = Vec::new();
|
let mut responses = Vec::new();
|
||||||
for object in objects {
|
for object in objects {
|
||||||
let path = format!("{}/{}", req.path(), object.get_id());
|
let path = format!("{}/{}.vcf", req.path(), object.get_id());
|
||||||
responses.push(
|
responses.push(
|
||||||
AddressObjectResource {
|
AddressObjectResource {
|
||||||
object,
|
object,
|
||||||
|
|||||||
@@ -1,19 +1,19 @@
|
|||||||
use crate::{
|
use crate::{
|
||||||
address_object::resource::{AddressObjectPropWrapper, AddressObjectResource},
|
|
||||||
Error,
|
Error,
|
||||||
|
address_object::resource::{AddressObjectPropWrapper, AddressObjectResource},
|
||||||
};
|
};
|
||||||
use actix_web::{http::StatusCode, HttpRequest};
|
use actix_web::{HttpRequest, http::StatusCode};
|
||||||
use rustical_dav::{
|
use rustical_dav::{
|
||||||
resource::Resource,
|
resource::Resource,
|
||||||
xml::{
|
xml::{
|
||||||
multistatus::ResponseElement, sync_collection::SyncCollectionRequest, MultistatusElement,
|
MultistatusElement, PropElement, PropfindType, multistatus::ResponseElement,
|
||||||
PropElement, PropfindType,
|
sync_collection::SyncCollectionRequest,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
use rustical_store::{
|
use rustical_store::{
|
||||||
|
AddressbookStore,
|
||||||
auth::User,
|
auth::User,
|
||||||
synctoken::{format_synctoken, parse_synctoken},
|
synctoken::{format_synctoken, parse_synctoken},
|
||||||
AddressbookStore,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
pub async fn handle_sync_collection<AS: AddressbookStore>(
|
pub async fn handle_sync_collection<AS: AddressbookStore>(
|
||||||
@@ -44,7 +44,11 @@ pub async fn handle_sync_collection<AS: AddressbookStore>(
|
|||||||
|
|
||||||
let mut responses = Vec::new();
|
let mut responses = Vec::new();
|
||||||
for object in new_objects {
|
for object in new_objects {
|
||||||
let path = format!("{}/{}", req.path().trim_end_matches('/'), object.get_id());
|
let path = format!(
|
||||||
|
"{}/{}.vcf",
|
||||||
|
req.path().trim_end_matches('/'),
|
||||||
|
object.get_id()
|
||||||
|
);
|
||||||
responses.push(
|
responses.push(
|
||||||
AddressObjectResource {
|
AddressObjectResource {
|
||||||
object,
|
object,
|
||||||
@@ -55,7 +59,7 @@ pub async fn handle_sync_collection<AS: AddressbookStore>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
for object_id in deleted_objects {
|
for object_id in deleted_objects {
|
||||||
let path = format!("{}/{}", req.path().trim_end_matches('/'), object_id);
|
let path = format!("{}/{}.vcf", req.path().trim_end_matches('/'), object_id);
|
||||||
responses.push(ResponseElement {
|
responses.push(ResponseElement {
|
||||||
href: path,
|
href: path,
|
||||||
status: Some(StatusCode::NOT_FOUND),
|
status: Some(StatusCode::NOT_FOUND),
|
||||||
|
|||||||
@@ -68,13 +68,13 @@ pub fn carddav_service<AP: AuthenticationProvider, A: AddressbookStore, S: Subsc
|
|||||||
.name(PrincipalResource::route_name()),
|
.name(PrincipalResource::route_name()),
|
||||||
)
|
)
|
||||||
.service(
|
.service(
|
||||||
web::scope("/{addressbook}")
|
web::scope("/{addressbook_id}")
|
||||||
.service(
|
.service(
|
||||||
AddressbookResourceService::<A, S>::new(store.clone())
|
AddressbookResourceService::<A, S>::new(store.clone())
|
||||||
.actix_resource(),
|
.actix_resource(),
|
||||||
)
|
)
|
||||||
.service(
|
.service(
|
||||||
web::scope("/{object}").service(
|
web::scope("/{object_id}.vcf").service(
|
||||||
AddressObjectResourceService::<A>::new(store.clone())
|
AddressObjectResourceService::<A>::new(store.clone())
|
||||||
.actix_resource(),
|
.actix_resource(),
|
||||||
),
|
),
|
||||||
|
|||||||
Reference in New Issue
Block a user