/* UIxMailPartICalActions.m - this file is part of SOGo * * Copyright (C) 2007 Inverse groupe conseil * * Author: Wolfgang Sourdeau * * This file is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This file is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 59 Temple Place - Suite 330, * Boston, MA 02111-1307, USA. */ #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import "UIxMailPartICalActions.h" @implementation UIxMailPartICalActions - (iCalEvent *) _emailEvent { NSData *content; NSString *eventString; iCalCalendar *emailCalendar; content = [[self clientObject] fetchBLOB]; eventString = [[NSString alloc] initWithData: content encoding: NSUTF8StringEncoding]; if (!eventString) eventString = [[NSString alloc] initWithData: content encoding: NSISOLatin1StringEncoding]; emailCalendar = [iCalCalendar parseSingleFromSource: eventString]; return (iCalEvent *) [emailCalendar firstChildWithTag: @"vevent"]; } - (SOGoAppointmentObject *) _eventObjectWithUID: (NSString *) uid forUser: (SOGoUser *) user { SOGoAppointmentFolder *personalFolder; SOGoAppointmentObject *eventObject; NSString *cname; eventObject = nil; personalFolder = [user personalCalendarFolderInContext: context]; cname = [personalFolder resourceNameForEventUID: uid]; if (cname) { eventObject = [personalFolder lookupName: cname inContext: context acquire: NO]; if (![eventObject isKindOfClass: [SOGoAppointmentObject class]]) eventObject = nil; } if (!eventObject) eventObject = [SOGoAppointmentObject objectWithName: uid inContainer: personalFolder]; return eventObject; } - (SOGoAppointmentObject *) _eventObjectWithUID: (NSString *) uid { return [self _eventObjectWithUID: uid forUser: [context activeUser]]; } - (void) _fixOrganizerInEvent: (iCalEvent *) brokenEvent { iCalPerson *organizer; SOGoMailObject *mail; NSArray *addresses; NGImap4EnvelopeAddress *from; id value; organizer = nil; mail = [[self clientObject] mailObject]; addresses = [mail replyToEnvelopeAddresses]; if (![addresses count]) addresses = [mail fromEnvelopeAddresses]; if ([addresses count] > 0) { from = [addresses objectAtIndex: 0]; value = [from baseEMail]; if ([value isNotNull]) { organizer = [iCalPerson elementWithTag: @"organizer"]; [organizer setEmail: value]; value = [from personalName]; if ([value isNotNull]) [organizer setCn: value]; [brokenEvent setOrganizer: organizer]; } } if (!organizer) [self errorWithFormat: @"no organizer could be found, SOGo will crash" @" if the user replies"]; } - (iCalEvent *) _setupChosenEventAndEventObject: (SOGoAppointmentObject **) eventObject { iCalEvent *emailEvent, *calendarEvent, *chosenEvent; iCalPerson *organizer; emailEvent = [self _emailEvent]; if (emailEvent) { *eventObject = [self _eventObjectWithUID: [emailEvent uid]]; if ([*eventObject isNew]) chosenEvent = emailEvent; else { calendarEvent = (iCalEvent *) [*eventObject component: NO secure: NO]; if ([calendarEvent compare: emailEvent] == NSOrderedAscending) chosenEvent = emailEvent; else chosenEvent = calendarEvent; } organizer = [chosenEvent organizer]; if (![[organizer rfc822Email] length]) [self _fixOrganizerInEvent: chosenEvent]; } else chosenEvent = nil; return chosenEvent; } #warning this is code copied from SOGoAppointmentObject... - (void) _updateAttendee: (iCalPerson *) attendee withSequence: (NSNumber *) sequence andCalUID: (NSString *) calUID forUID: (NSString *) uid { SOGoAppointmentObject *eventObject; iCalEvent *event; iCalPerson *otherAttendee; NSString *iCalString; eventObject = [self _eventObjectWithUID: calUID forUser: [SOGoUser userWithLogin: uid roles: nil]]; if (![eventObject isNew]) { event = [eventObject component: NO secure: NO]; if ([[event sequence] compare: sequence] == NSOrderedSame) { otherAttendee = [event findParticipantWithEmail: [attendee rfc822Email]]; [otherAttendee setPartStat: [attendee partStat]]; iCalString = [[event parent] versitString]; [eventObject saveContentString: iCalString]; } } } - (WOResponse *) _changePartStatusAction: (NSString *) newStatus { WOResponse *response; SOGoAppointmentObject *eventObject; iCalEvent *chosenEvent; iCalPerson *user; iCalCalendar *emailCalendar, *calendar; NSString *rsvp, *method, *organizerUID; chosenEvent = [self _setupChosenEventAndEventObject: &eventObject]; if (chosenEvent) { user = [chosenEvent findParticipant: [context activeUser]]; [user setPartStat: newStatus]; calendar = [chosenEvent parent]; emailCalendar = [[self _emailEvent] parent]; method = [[emailCalendar method] lowercaseString]; if ([method isEqualToString: @"request"]) { [calendar setMethod: @""]; rsvp = [[user rsvp] lowercaseString]; } else rsvp = nil; [eventObject saveContentString: [calendar versitString]]; if ([rsvp isEqualToString: @"true"]) [eventObject sendResponseToOrganizer]; organizerUID = [[chosenEvent organizer] uid]; if (organizerUID) [self _updateAttendee: user withSequence: [chosenEvent sequence] andCalUID: [chosenEvent uid] forUID: organizerUID]; response = [self responseWith204]; } else { response = [context response]; [response setStatus: 409]; } return response; } - (WOResponse *) acceptAction { return [self _changePartStatusAction: @"ACCEPTED"]; } - (WOResponse *) declineAction { return [self _changePartStatusAction: @"DECLINED"]; } - (WOResponse *) deleteFromCalendarAction { iCalEvent *emailEvent; SOGoAppointmentObject *eventObject; WOResponse *response; emailEvent = [self _emailEvent]; if (emailEvent) { eventObject = [self _eventObjectWithUID: [emailEvent uid]]; [eventObject delete]; response = [self responseWith204]; } else { response = [context response]; [response setStatus: 409]; } return response; } - (iCalPerson *) _emailParticipantWithEvent: (iCalEvent *) event { NSString *emailFrom; SOGoMailObject *mailObject; NGImap4EnvelopeAddress *address; mailObject = [[self clientObject] mailObject]; address = [[mailObject fromEnvelopeAddresses] objectAtIndex: 0]; emailFrom = [address baseEMail]; return [event findParticipantWithEmail: emailFrom]; } - (BOOL) _updateParticipantStatusInEvent: (iCalEvent *) calendarEvent fromEvent: (iCalEvent *) emailEvent inObject: (SOGoAppointmentObject *) eventObject { iCalPerson *calendarParticipant, *mailParticipant; NSString *partStat; BOOL result; calendarParticipant = [self _emailParticipantWithEvent: calendarEvent]; mailParticipant = [self _emailParticipantWithEvent: emailEvent]; if (calendarParticipant && mailParticipant) { result = YES; partStat = [mailParticipant partStat]; if ([partStat caseInsensitiveCompare: [calendarParticipant partStat]] != NSOrderedSame) { [calendarParticipant setPartStat: [partStat uppercaseString]]; [eventObject saveComponent: calendarEvent]; } } else result = NO; return result; } - (WOResponse *) updateUserStatusAction { iCalEvent *emailEvent, *calendarEvent; SOGoAppointmentObject *eventObject; WOResponse *response; response = nil; emailEvent = [self _emailEvent]; if (emailEvent) { eventObject = [self _eventObjectWithUID: [emailEvent uid]]; calendarEvent = [eventObject component: NO secure: NO]; if (([[emailEvent sequence] compare: [calendarEvent sequence]] != NSOrderedDescending) && ([self _updateParticipantStatusInEvent: calendarEvent fromEvent: emailEvent inObject: eventObject])) response = [self responseWith204]; } if (!response) { response = [context response]; [response setStatus: 409]; } return response; } // - (WOResponse *) markTentativeAction // { // return [self _changePartStatusAction: @"TENTATIVE"]; // } // - (WOResponse *) addToCalendarAction // { // return [self responseWithStatus: 404]; // } // - (WOResponse *) deleteFromCalendarAction // { // return [self responseWithStatus: 404]; // } @end