583 lines
19 KiB
Rust
583 lines
19 KiB
Rust
use axum::{
|
|
extract::{Path, State},
|
|
http::StatusCode,
|
|
response::{IntoResponse, Json},
|
|
routing::{get, post},
|
|
Router,
|
|
};
|
|
use chrono::{DateTime, NaiveDateTime, TimeZone, Utc};
|
|
use diesel::r2d2::{ConnectionManager, Pool};
|
|
use diesel::PgConnection;
|
|
use icalendar::{
|
|
Calendar, CalendarDateTime, Component, DatePerhapsTime, Event as IcalEvent, EventLike, Property,
|
|
};
|
|
use log::info;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::collections::HashMap;
|
|
use std::sync::Arc;
|
|
use tokio::sync::RwLock;
|
|
use uuid::Uuid;
|
|
|
|
use crate::core::urls::ApiUrls;
|
|
use crate::shared::state::AppState;
|
|
|
|
pub mod caldav;
|
|
|
|
pub struct CalendarState {
|
|
events: RwLock<HashMap<Uuid, CalendarEvent>>,
|
|
}
|
|
|
|
impl CalendarState {
|
|
pub fn new() -> Self {
|
|
Self {
|
|
events: RwLock::new(HashMap::new()),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Default for CalendarState {
|
|
fn default() -> Self {
|
|
Self::new()
|
|
}
|
|
}
|
|
|
|
static CALENDAR_STATE: std::sync::OnceLock<CalendarState> = std::sync::OnceLock::new();
|
|
|
|
fn get_calendar_state() -> &'static CalendarState {
|
|
CALENDAR_STATE.get_or_init(CalendarState::new)
|
|
}
|
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
pub struct CalendarEvent {
|
|
pub id: Uuid,
|
|
pub title: String,
|
|
pub description: Option<String>,
|
|
pub start_time: DateTime<Utc>,
|
|
pub end_time: DateTime<Utc>,
|
|
pub location: Option<String>,
|
|
pub attendees: Vec<String>,
|
|
pub organizer: String,
|
|
pub reminder_minutes: Option<i32>,
|
|
pub recurrence: Option<String>,
|
|
pub created_at: DateTime<Utc>,
|
|
pub updated_at: DateTime<Utc>,
|
|
}
|
|
|
|
#[derive(Debug, Clone, Serialize, Deserialize)]
|
|
pub struct CalendarEventInput {
|
|
pub title: String,
|
|
pub description: Option<String>,
|
|
pub start_time: DateTime<Utc>,
|
|
pub end_time: DateTime<Utc>,
|
|
pub location: Option<String>,
|
|
#[serde(default)]
|
|
pub attendees: Vec<String>,
|
|
pub organizer: String,
|
|
pub reminder_minutes: Option<i32>,
|
|
pub recurrence: Option<String>,
|
|
}
|
|
|
|
impl CalendarEvent {
|
|
pub fn to_ical(&self) -> IcalEvent {
|
|
let mut event = IcalEvent::new();
|
|
event.uid(&self.id.to_string());
|
|
event.summary(&self.title);
|
|
event.starts(self.start_time);
|
|
event.ends(self.end_time);
|
|
|
|
if let Some(ref desc) = self.description {
|
|
event.description(desc);
|
|
}
|
|
if let Some(ref loc) = self.location {
|
|
event.location(loc);
|
|
}
|
|
|
|
event.add_property("ORGANIZER", format!("mailto:{}", self.organizer));
|
|
|
|
for attendee in &self.attendees {
|
|
event.add_property("ATTENDEE", format!("mailto:{}", attendee));
|
|
}
|
|
|
|
if let Some(ref rrule) = self.recurrence {
|
|
event.add_property("RRULE", rrule);
|
|
}
|
|
|
|
if let Some(minutes) = self.reminder_minutes {
|
|
event.add_property("VALARM", format!("-PT{}M", minutes));
|
|
}
|
|
|
|
event.done()
|
|
}
|
|
|
|
pub fn from_ical(ical: &IcalEvent, organizer: &str) -> Option<Self> {
|
|
let uid = ical.get_uid()?;
|
|
let summary = ical.get_summary()?;
|
|
|
|
let start_time = date_perhaps_time_to_utc(ical.get_start()?)?;
|
|
let end_time = date_perhaps_time_to_utc(ical.get_end()?)?;
|
|
|
|
let id = Uuid::parse_str(uid).unwrap_or_else(|_| Uuid::new_v4());
|
|
|
|
Some(Self {
|
|
id,
|
|
title: summary.to_string(),
|
|
description: ical.get_description().map(String::from),
|
|
start_time,
|
|
end_time,
|
|
location: ical.get_location().map(String::from),
|
|
attendees: Vec::new(),
|
|
organizer: organizer.to_string(),
|
|
reminder_minutes: None,
|
|
recurrence: None,
|
|
created_at: Utc::now(),
|
|
updated_at: Utc::now(),
|
|
})
|
|
}
|
|
}
|
|
|
|
fn date_perhaps_time_to_utc(dpt: DatePerhapsTime) -> Option<DateTime<Utc>> {
|
|
match dpt {
|
|
DatePerhapsTime::DateTime(cal_dt) => match cal_dt {
|
|
CalendarDateTime::Utc(dt) => Some(dt),
|
|
CalendarDateTime::Floating(naive) => Some(Utc.from_utc_datetime(&naive)),
|
|
CalendarDateTime::WithTimezone { date_time, .. } => {
|
|
Some(Utc.from_utc_datetime(&date_time))
|
|
}
|
|
},
|
|
DatePerhapsTime::Date(date) => {
|
|
let naive = NaiveDateTime::new(date, chrono::NaiveTime::from_hms_opt(0, 0, 0)?);
|
|
Some(Utc.from_utc_datetime(&naive))
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn export_to_ical(events: &[CalendarEvent], calendar_name: &str) -> String {
|
|
let mut calendar = Calendar::new();
|
|
calendar.name(calendar_name);
|
|
calendar.append_property(Property::new("PRODID", "-//GeneralBots//Calendar//EN"));
|
|
|
|
for event in events {
|
|
calendar.push(event.to_ical());
|
|
}
|
|
|
|
calendar.done().to_string()
|
|
}
|
|
|
|
pub fn import_from_ical(ical_str: &str, organizer: &str) -> Vec<CalendarEvent> {
|
|
let Ok(calendar) = ical_str.parse::<Calendar>() else {
|
|
return Vec::new();
|
|
};
|
|
|
|
calendar
|
|
.components
|
|
.iter()
|
|
.filter_map(|c| {
|
|
if let icalendar::CalendarComponent::Event(e) = c {
|
|
CalendarEvent::from_ical(e, organizer)
|
|
} else {
|
|
None
|
|
}
|
|
})
|
|
.collect()
|
|
}
|
|
|
|
#[derive(Default)]
|
|
pub struct CalendarEngine {
|
|
events: Vec<CalendarEvent>,
|
|
#[allow(dead_code)]
|
|
conn: Option<Pool<ConnectionManager<PgConnection>>>,
|
|
}
|
|
|
|
impl CalendarEngine {
|
|
pub fn new(conn: Pool<ConnectionManager<PgConnection>>) -> Self {
|
|
Self {
|
|
events: Vec::new(),
|
|
conn: Some(conn),
|
|
}
|
|
}
|
|
|
|
pub fn create_event(&mut self, input: CalendarEventInput) -> CalendarEvent {
|
|
let event = CalendarEvent {
|
|
id: Uuid::new_v4(),
|
|
title: input.title,
|
|
description: input.description,
|
|
start_time: input.start_time,
|
|
end_time: input.end_time,
|
|
location: input.location,
|
|
attendees: input.attendees,
|
|
organizer: input.organizer,
|
|
reminder_minutes: input.reminder_minutes,
|
|
recurrence: input.recurrence,
|
|
created_at: Utc::now(),
|
|
updated_at: Utc::now(),
|
|
};
|
|
self.events.push(event.clone());
|
|
event
|
|
}
|
|
|
|
pub fn get_event(&self, id: Uuid) -> Option<&CalendarEvent> {
|
|
self.events.iter().find(|e| e.id == id)
|
|
}
|
|
|
|
pub fn update_event(&mut self, id: Uuid, input: CalendarEventInput) -> Option<CalendarEvent> {
|
|
let event = self.events.iter_mut().find(|e| e.id == id)?;
|
|
event.title = input.title;
|
|
event.description = input.description;
|
|
event.start_time = input.start_time;
|
|
event.end_time = input.end_time;
|
|
event.location = input.location;
|
|
event.attendees = input.attendees;
|
|
event.organizer = input.organizer;
|
|
event.reminder_minutes = input.reminder_minutes;
|
|
event.recurrence = input.recurrence;
|
|
event.updated_at = Utc::now();
|
|
Some(event.clone())
|
|
}
|
|
|
|
pub fn delete_event(&mut self, id: Uuid) -> bool {
|
|
let len = self.events.len();
|
|
self.events.retain(|e| e.id != id);
|
|
self.events.len() < len
|
|
}
|
|
|
|
pub fn list_events(&self, limit: usize, offset: usize) -> Vec<&CalendarEvent> {
|
|
self.events.iter().skip(offset).take(limit).collect()
|
|
}
|
|
|
|
pub fn get_events_range(
|
|
&self,
|
|
start: DateTime<Utc>,
|
|
end: DateTime<Utc>,
|
|
) -> Vec<&CalendarEvent> {
|
|
self.events
|
|
.iter()
|
|
.filter(|e| e.start_time >= start && e.end_time <= end)
|
|
.collect()
|
|
}
|
|
|
|
pub fn get_user_events(&self, user_id: &str) -> Vec<&CalendarEvent> {
|
|
self.events
|
|
.iter()
|
|
.filter(|e| e.organizer == user_id)
|
|
.collect()
|
|
}
|
|
|
|
pub fn check_conflicts(
|
|
&self,
|
|
start: DateTime<Utc>,
|
|
end: DateTime<Utc>,
|
|
user_id: &str,
|
|
) -> Vec<&CalendarEvent> {
|
|
self.events
|
|
.iter()
|
|
.filter(|e| e.organizer == user_id && e.start_time < end && e.end_time > start)
|
|
.collect()
|
|
}
|
|
|
|
pub fn export_ical(&self, calendar_name: &str) -> String {
|
|
export_to_ical(&self.events, calendar_name)
|
|
}
|
|
|
|
pub fn import_ical(&mut self, ical_str: &str, organizer: &str) -> usize {
|
|
let imported = import_from_ical(ical_str, organizer);
|
|
let count = imported.len();
|
|
self.events.extend(imported);
|
|
count
|
|
}
|
|
}
|
|
|
|
pub async fn list_events(
|
|
State(_state): State<Arc<AppState>>,
|
|
axum::extract::Query(_query): axum::extract::Query<serde_json::Value>,
|
|
) -> Json<Vec<CalendarEvent>> {
|
|
let calendar_state = get_calendar_state();
|
|
let events = calendar_state.events.read().await;
|
|
|
|
let mut result: Vec<CalendarEvent> = events.values().cloned().collect();
|
|
result.sort_by(|a, b| a.start_time.cmp(&b.start_time));
|
|
|
|
Json(result)
|
|
}
|
|
|
|
pub async fn list_calendars_api(State(_state): State<Arc<AppState>>) -> Json<serde_json::Value> {
|
|
Json(serde_json::json!({
|
|
"calendars": [
|
|
{
|
|
"id": "default",
|
|
"name": "My Calendar",
|
|
"color": "#3b82f6",
|
|
"visible": true
|
|
}
|
|
]
|
|
}))
|
|
}
|
|
|
|
pub async fn list_calendars(State(_state): State<Arc<AppState>>) -> axum::response::Html<String> {
|
|
axum::response::Html(r#"
|
|
<div class="calendar-item" data-calendar-id="default">
|
|
<span class="calendar-checkbox checked" style="background: #3b82f6;" onclick="toggleCalendar(this)"></span>
|
|
<span class="calendar-name">My Calendar</span>
|
|
</div>
|
|
<div class="calendar-item" data-calendar-id="work">
|
|
<span class="calendar-checkbox checked" style="background: #22c55e;" onclick="toggleCalendar(this)"></span>
|
|
<span class="calendar-name">Work</span>
|
|
</div>
|
|
<div class="calendar-item" data-calendar-id="personal">
|
|
<span class="calendar-checkbox checked" style="background: #f59e0b;" onclick="toggleCalendar(this)"></span>
|
|
<span class="calendar-name">Personal</span>
|
|
</div>
|
|
"#.to_string())
|
|
}
|
|
|
|
pub async fn upcoming_events_api(State(_state): State<Arc<AppState>>) -> Json<serde_json::Value> {
|
|
Json(serde_json::json!({
|
|
"events": [],
|
|
"message": "No upcoming events"
|
|
}))
|
|
}
|
|
|
|
pub async fn upcoming_events(State(_state): State<Arc<AppState>>) -> axum::response::Html<String> {
|
|
axum::response::Html(
|
|
r#"
|
|
<div class="upcoming-event">
|
|
<div class="upcoming-color" style="background: #3b82f6;"></div>
|
|
<div class="upcoming-info">
|
|
<span class="upcoming-title">No upcoming events</span>
|
|
<span class="upcoming-time">Create your first event</span>
|
|
</div>
|
|
</div>
|
|
"#
|
|
.to_string(),
|
|
)
|
|
}
|
|
|
|
pub async fn get_event(
|
|
State(_state): State<Arc<AppState>>,
|
|
Path(id): Path<Uuid>,
|
|
) -> Result<Json<CalendarEvent>, StatusCode> {
|
|
let calendar_state = get_calendar_state();
|
|
let events = calendar_state.events.read().await;
|
|
|
|
events
|
|
.get(&id)
|
|
.cloned()
|
|
.map(Json)
|
|
.ok_or(StatusCode::NOT_FOUND)
|
|
}
|
|
|
|
pub async fn create_event(
|
|
State(_state): State<Arc<AppState>>,
|
|
Json(input): Json<CalendarEventInput>,
|
|
) -> Result<Json<CalendarEvent>, StatusCode> {
|
|
let calendar_state = get_calendar_state();
|
|
let now = Utc::now();
|
|
|
|
let event = CalendarEvent {
|
|
id: Uuid::new_v4(),
|
|
title: input.title,
|
|
description: input.description,
|
|
start_time: input.start_time,
|
|
end_time: input.end_time,
|
|
location: input.location,
|
|
attendees: input.attendees,
|
|
organizer: input.organizer,
|
|
reminder_minutes: input.reminder_minutes,
|
|
recurrence: input.recurrence,
|
|
created_at: now,
|
|
updated_at: now,
|
|
};
|
|
|
|
let mut events = calendar_state.events.write().await;
|
|
events.insert(event.id, event.clone());
|
|
|
|
log::info!("Created calendar event: {} ({})", event.title, event.id);
|
|
|
|
Ok(Json(event))
|
|
}
|
|
|
|
pub async fn update_event(
|
|
State(_state): State<Arc<AppState>>,
|
|
Path(id): Path<Uuid>,
|
|
Json(input): Json<CalendarEventInput>,
|
|
) -> Result<Json<CalendarEvent>, StatusCode> {
|
|
let calendar_state = get_calendar_state();
|
|
let mut events = calendar_state.events.write().await;
|
|
|
|
let event = events.get_mut(&id).ok_or(StatusCode::NOT_FOUND)?;
|
|
|
|
event.title = input.title;
|
|
event.description = input.description;
|
|
event.start_time = input.start_time;
|
|
event.end_time = input.end_time;
|
|
event.location = input.location;
|
|
event.attendees = input.attendees;
|
|
event.organizer = input.organizer;
|
|
event.reminder_minutes = input.reminder_minutes;
|
|
event.recurrence = input.recurrence;
|
|
event.updated_at = Utc::now();
|
|
|
|
log::info!("Updated calendar event: {} ({})", event.title, event.id);
|
|
|
|
Ok(Json(event.clone()))
|
|
}
|
|
|
|
pub async fn delete_event(State(_state): State<Arc<AppState>>, Path(id): Path<Uuid>) -> StatusCode {
|
|
let calendar_state = get_calendar_state();
|
|
let mut events = calendar_state.events.write().await;
|
|
|
|
if events.remove(&id).is_some() {
|
|
log::info!("Deleted calendar event: {}", id);
|
|
StatusCode::NO_CONTENT
|
|
} else {
|
|
StatusCode::NOT_FOUND
|
|
}
|
|
}
|
|
|
|
pub async fn export_ical(State(_state): State<Arc<AppState>>) -> impl IntoResponse {
|
|
let calendar = Calendar::new().name("GeneralBots Calendar").done();
|
|
(
|
|
[(
|
|
axum::http::header::CONTENT_TYPE,
|
|
"text/calendar; charset=utf-8",
|
|
)],
|
|
calendar.to_string(),
|
|
)
|
|
}
|
|
|
|
pub async fn import_ical(
|
|
State(_state): State<Arc<AppState>>,
|
|
body: String,
|
|
) -> Result<Json<serde_json::Value>, StatusCode> {
|
|
let events = import_from_ical(&body, "unknown");
|
|
Ok(Json(serde_json::json!({ "imported": events.len() })))
|
|
}
|
|
|
|
pub async fn new_event_form(State(_state): State<Arc<AppState>>) -> axum::response::Html<String> {
|
|
axum::response::Html(
|
|
r#"
|
|
<div class="event-form-content">
|
|
<p>Create a new event using the form on the right panel.</p>
|
|
</div>
|
|
"#
|
|
.to_string(),
|
|
)
|
|
}
|
|
|
|
pub async fn new_calendar_form(
|
|
State(_state): State<Arc<AppState>>,
|
|
) -> axum::response::Html<String> {
|
|
axum::response::Html(r##"
|
|
<form class="calendar-form" hx-post="/api/calendar/calendars" hx-swap="none">
|
|
<div class="form-group">
|
|
<label>Calendar Name</label>
|
|
<input type="text" name="name" placeholder="My Calendar" required />
|
|
</div>
|
|
<div class="form-group">
|
|
<label>Color</label>
|
|
<div class="color-options">
|
|
<label><input type="radio" name="color" value="#3b82f6" checked /><span class="color-dot" style="background:#3b82f6"></span></label>
|
|
<label><input type="radio" name="color" value="#22c55e" /><span class="color-dot" style="background:#22c55e"></span></label>
|
|
<label><input type="radio" name="color" value="#f59e0b" /><span class="color-dot" style="background:#f59e0b"></span></label>
|
|
<label><input type="radio" name="color" value="#ef4444" /><span class="color-dot" style="background:#ef4444"></span></label>
|
|
<label><input type="radio" name="color" value="#8b5cf6" /><span class="color-dot" style="background:#8b5cf6"></span></label>
|
|
</div>
|
|
</div>
|
|
<div class="form-actions">
|
|
<button type="button" class="btn-secondary" onclick="this.closest('.modal').classList.add('hidden')">Cancel</button>
|
|
<button type="submit" class="btn-primary">Create Calendar</button>
|
|
</div>
|
|
</form>
|
|
"##.to_string())
|
|
}
|
|
|
|
pub async fn start_reminder_job(engine: Arc<CalendarEngine>) {
|
|
info!("Starting calendar reminder job");
|
|
|
|
loop {
|
|
tokio::time::sleep(tokio::time::Duration::from_secs(60)).await;
|
|
|
|
let now = Utc::now();
|
|
for event in &engine.events {
|
|
if let Some(reminder_minutes) = event.reminder_minutes {
|
|
let reminder_time =
|
|
event.start_time - chrono::Duration::minutes(i64::from(reminder_minutes));
|
|
|
|
if now >= reminder_time && now < reminder_time + chrono::Duration::minutes(1) {
|
|
info!(
|
|
"Reminder: Event '{}' starts in {} minutes",
|
|
event.title, reminder_minutes
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn configure_calendar_routes() -> Router<Arc<AppState>> {
|
|
Router::new()
|
|
.route(
|
|
ApiUrls::CALENDAR_EVENTS,
|
|
get(list_events).post(create_event),
|
|
)
|
|
.route(
|
|
&ApiUrls::CALENDAR_EVENT_BY_ID.replace(":id", "{id}"),
|
|
get(get_event).put(update_event).delete(delete_event),
|
|
)
|
|
.route(ApiUrls::CALENDAR_EXPORT, get(export_ical))
|
|
.route(ApiUrls::CALENDAR_IMPORT, post(import_ical))
|
|
.route(ApiUrls::CALENDAR_CALENDARS, get(list_calendars_api))
|
|
.route(ApiUrls::CALENDAR_UPCOMING, get(upcoming_events_api))
|
|
.route("/ui/calendar/list", get(list_calendars))
|
|
.route("/ui/calendar/upcoming", get(upcoming_events))
|
|
.route("/ui/calendar/event/new", get(new_event_form))
|
|
.route("/ui/calendar/new", get(new_calendar_form))
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use chrono::Utc;
|
|
use uuid::Uuid;
|
|
|
|
#[test]
|
|
fn test_event_to_ical_roundtrip() {
|
|
let event = CalendarEvent {
|
|
id: Uuid::new_v4(),
|
|
title: "Test Meeting".to_string(),
|
|
description: Some("A test meeting".to_string()),
|
|
start_time: Utc::now(),
|
|
end_time: Utc::now() + chrono::Duration::hours(1),
|
|
location: Some("Room 101".to_string()),
|
|
attendees: vec!["user@example.com".to_string()],
|
|
organizer: "organizer@example.com".to_string(),
|
|
reminder_minutes: Some(15),
|
|
recurrence: None,
|
|
created_at: Utc::now(),
|
|
updated_at: Utc::now(),
|
|
};
|
|
|
|
let ical = event.to_ical();
|
|
assert_eq!(ical.get_summary(), Some("Test Meeting"));
|
|
assert_eq!(ical.get_location(), Some("Room 101"));
|
|
}
|
|
|
|
#[test]
|
|
fn test_export_import_ical() {
|
|
let mut engine = CalendarEngine::default();
|
|
engine.create_event(CalendarEventInput {
|
|
title: "Event 1".to_string(),
|
|
description: None,
|
|
start_time: Utc::now(),
|
|
end_time: Utc::now() + chrono::Duration::hours(1),
|
|
location: None,
|
|
attendees: vec![],
|
|
organizer: "test@example.com".to_string(),
|
|
reminder_minutes: None,
|
|
recurrence: None,
|
|
});
|
|
|
|
let ical = engine.export_ical("Test Calendar");
|
|
assert!(ical.contains("BEGIN:VCALENDAR"));
|
|
assert!(ical.contains("Event 1"));
|
|
}
|
|
}
|