botserver/src/calendar/mod.rs

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"));
}
}