/* UIxMailAccountActions.m - this file is part of SOGo * * Copyright (C) 2007-2020 Inverse inc. * * 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. */ #if defined(HAVE_OPENSSL) || defined(HAVE_GNUTLS) #include #include #include #include #endif #import #import #import #import #import #define COMPILING_NGOBJWEB 1 /* httpRequest is needed in importCertificateAction */ #import #undef COMPILING_NGOBJWEB #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import #import "UIxMailAccountActions.h" @implementation UIxMailAccountActions - (WOResponse *) listMailboxesAction { SOGoMailAccount *co; NSArray *folders; NSDictionary *data; co = [self clientObject]; folders = [co allFoldersMetadata: SOGoMailStandardListing]; data = [NSDictionary dictionaryWithObjectsAndKeys: folders, @"mailboxes", [co getInboxQuota], @"quotas", nil]; return [self responseWithStatus: 200 andJSONRepresentation: data]; } - (WOResponse *) listAllMailboxesAction { SOGoMailAccount *co; NSArray *folders; NSDictionary *data; co = [self clientObject]; folders = [co allFoldersMetadata: SOGoMailSubscriptionsManagementListing]; data = [NSDictionary dictionaryWithObjectsAndKeys: folders, @"mailboxes", nil]; return [self responseWithStatus: 200 andJSONRepresentation: data]; } /* compose */ - (NSString *) _emailFromIdentity: (NSDictionary *) identity { NSString *fullName, *format; fullName = [identity objectForKey: @"fullName"]; if ([fullName length]) format = @"%{fullName} <%{email}>"; else format = @"%{email}"; return [identity keysWithFormat: format]; } - (WOResponse *) composeAction { BOOL save, isHTML; NSDictionary *data, *identity; NSMutableDictionary *headers; NSString *accountName, *mailboxName, *messageName; NSString *value, *signature, *nl, *space; SOGoDraftObject *newDraftMessage; SOGoDraftsFolder *drafts; SOGoMailAccount *co; SOGoUserDefaults *ud; id mailTo; co = [self clientObject]; drafts = [co draftsFolderInContext: context]; newDraftMessage = [drafts newDraft]; headers = [NSMutableDictionary dictionary]; save = NO; value = [[self request] formValueForKey: @"mailto"]; if ([value length] > 0) { mailTo = [[value stringByUnescapingURL] objectFromJSONString]; if (mailTo && [mailTo isKindOfClass: [NSArray class]]) { [headers setObject: (NSArray *) mailTo forKey: @"to"]; save = YES; } } value = [[self request] formValueForKey: @"subject"]; if ([value length] > 0) { [headers setObject: [value stringByUnescapingURL] forKey: @"subject"]; save = YES; } identity = [co defaultIdentity]; if (identity) { [headers setObject: [self _emailFromIdentity: identity] forKey: @"from"]; signature = [identity objectForKey: @"signature"]; if ([signature length]) { ud = [[context activeUser] userDefaults]; [newDraftMessage setIsHTML: [[ud mailComposeMessageType] isEqualToString: @"html"]]; isHTML = [newDraftMessage isHTML]; nl = (isHTML? @"
" : @"\n"); space = (isHTML ? @" " : @" "); [newDraftMessage setText: [NSString stringWithFormat: @"%@%@--%@%@%@", nl, nl, space, nl, signature]]; } save = YES; } if (save) { [newDraftMessage setHeaders: headers]; [newDraftMessage storeInfo]; } accountName = [co nameInContainer]; mailboxName = [drafts absoluteImap4Name]; // Ex: /INBOX/Drafts/ mailboxName = [mailboxName substringWithRange: NSMakeRange(1, [mailboxName length] -2)]; messageName = [newDraftMessage nameInContainer]; data = [NSDictionary dictionaryWithObjectsAndKeys: accountName, @"accountId", mailboxName, @"mailboxPath", messageName, @"draftId", nil]; return [self responseWithStatus: 201 andString: [data jsonRepresentation]]; } - (WOResponse *) _performDelegationAction: (SEL) action { SOGoMailAccount *co; WOResponse *response; NSString *uid; co = [self clientObject]; if ([[co nameInContainer] isEqualToString: @"0"]) { uid = [[context request] formValueForKey: @"uid"]; if ([uid length] > 0) { [co performSelector: action withObject: [NSArray arrayWithObject: uid]]; response = [self responseWith204]; } else response = [self responseWithStatus: 500 andString: @"Missing 'uid' parameter."]; } else response = [self responseWithStatus: 403 andString: @"This action cannot be performed on secondary accounts."]; return response; } - (WOResponse *) addDelegateAction { return [self _performDelegationAction: @selector (addDelegates:)]; } - (WOResponse *) removeDelegateAction { return [self _performDelegationAction: @selector (removeDelegates:)]; } - (WOResponse *) certificateAction { NSData *pem; NSDictionary *data; WOResponse *response; pem = [[self clientObject] certificate]; if (pem) { data = [pem certificateDescription]; if (data) { response = [self responseWithStatus: 200 andJSONRepresentation: data]; } else { data = [NSDictionary dictionaryWithObject: [self labelForKey: @"Error reading the certificate. Please install a new certificate."] forKey: @"message"]; response = [self responseWithStatus: 500 andJSONRepresentation: data]; } } else { data = [NSDictionary dictionaryWithObject: [self labelForKey: @"No certificate associated to account."] forKey: @"message"]; response = [self responseWithStatus: 404 andJSONRepresentation: data]; } return response; } - (WOResponse *) importCertificateAction { NSArray *parts; NGMimeBodyPart *part; NGMimeContentDispositionHeaderField *header; NSData *pkcs12; NSString *mimeType, *name, *password; WOResponse *response; id data; unsigned int count, max; password = nil; pkcs12 = nil; response = [self responseWithStatus: 507]; data = [[[context request] httpRequest] body]; if (![data isKindOfClass: [NSException class]]) { parts = [data parts]; max = [parts count]; for (count = 0; count < max; count++) { part = [parts objectAtIndex: count]; header = (NGMimeContentDispositionHeaderField *)[part headerForKey: @"content-disposition"]; name = [header name]; if ([name isEqualToString: @"password"]) { password = [part body]; } else if ([name isEqualToString: @"file"]) { mimeType = [(NGMimeType *)[part headerForKey: @"content-type"] stringValue]; if ([mimeType hasSuffix: @"pkcs12"]) { pkcs12 = [part body]; } else { response = [self responseWithStatus: 507]; } } } } if (password && pkcs12) { NSData *certificate; NSDictionary *description; certificate = [pkcs12 convertPKCS12ToPEMUsingPassword: password]; if (!certificate) return [self responseWithStatus: 507]; [[self clientObject] setCertificate: certificate]; description = [certificate certificateDescription]; if (description) { response = [self responseWithStatus: 200 andJSONRepresentation: description]; } else { description = [NSDictionary dictionaryWithObject: [self labelForKey: @"Error reading the certificate. Please install a new certificate."] forKey: @"message"]; response = [self responseWithStatus: 500 andJSONRepresentation: description]; } } return response; } - (WOResponse *) removeCertificateAction { [[self clientObject] setCertificate: nil]; return [self responseWith204]; } @end