/* Copyright (C) 2005 SKYRIX Software AG This file is part of OpenGroupware.org. OGo is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. OGo 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with OGo; see the file COPYING. If not, write to the Free Software Foundation, 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 "AgenorUserDefaults.h" #import "LDAPUserManager.h" #import "NSArray+Utilities.h" #import "SOGoCache.h" #import "SOGoDateFormatter.h" #import "SOGoObject.h" #import "SOGoPermissions.h" #import "SOGoUser.h" static NSTimeZone *serverTimeZone = nil; static NSString *fallbackIMAP4Server = nil; static BOOL fallbackIsConfigured = NO; static NSString *defaultLanguage = nil; static NSArray *superUsernames = nil; static NSURL *AgenorProfileURL = nil; static BOOL acceptAnyUser = NO; static int sogoFirstDayOfWeek = -1; static int defaultDayStartTime = -1; static int defaultDayEndTime = -1; NSString *SOGoWeekStartJanuary1 = @"January1"; NSString *SOGoWeekStartFirst4DayWeek = @"First4DayWeek"; NSString *SOGoWeekStartFirstFullWeek = @"FirstFullWeek"; @interface NSObject (SOGoRoles) - (NSArray *) rolesOfUser: (NSString *) uid; @end @implementation SoUser (SOGoExtension) - (NSString *) language { return [SOGoUser language]; } @end @implementation SOGoUser static int _timeValue (NSString *key) { int time; if (key && [key length] > 1) time = [[key substringToIndex: 2] intValue]; else time = -1; return time; } + (void) initialize { NSString *tzName; NSUserDefaults *ud; NSString *profileURL; ud = [NSUserDefaults standardUserDefaults]; if (!serverTimeZone) { tzName = [ud stringForKey: @"SOGoServerTimeZone"]; if (!tzName) tzName = @"UTC"; serverTimeZone = [NSTimeZone timeZoneWithName: tzName]; [serverTimeZone retain]; } if (sogoFirstDayOfWeek == -1) sogoFirstDayOfWeek = [ud integerForKey: @"SOGoFirstDayOfWeek"]; if (defaultDayStartTime == -1) { defaultDayStartTime = _timeValue ([ud stringForKey: @"SOGoDayStartTime"]); if (defaultDayStartTime == -1) defaultDayStartTime = 8; } if (defaultDayEndTime == -1) { defaultDayEndTime = _timeValue ([ud stringForKey: @"SOGoDayEndTime"]); if (defaultDayEndTime == -1) defaultDayEndTime = 18; } if (!AgenorProfileURL) { profileURL = [ud stringForKey: @"AgenorProfileURL"]; AgenorProfileURL = [[NSURL alloc] initWithString: profileURL]; } if (!fallbackIMAP4Server) ASSIGN (fallbackIMAP4Server, [ud stringForKey: @"SOGoFallbackIMAP4Server"]); if (fallbackIMAP4Server) fallbackIsConfigured = YES; else { [self warnWithFormat: @"no server specified for SOGoFallbackIMAP4Server," @" value set to 'localhost'"]; fallbackIMAP4Server = @"localhost"; } if (!defaultLanguage) { ASSIGN (defaultLanguage, [ud stringForKey: @"SOGoDefaultLanguage"]); if (!defaultLanguage) ASSIGN (defaultLanguage, @"English"); } if (!superUsernames) ASSIGN (superUsernames, [ud arrayForKey: @"SOGoSuperUsernames"]); acceptAnyUser = ([[ud stringForKey: @"SOGoAuthentificationMethod"] isEqualToString: @"bypass"]); } + (NSString *) language { NSArray *bLanguages; WOContext *context; NSString *lng; context = [[WOApplication application] context]; bLanguages = [[context request] browserLanguages]; if ([bLanguages count] > 0) lng = [bLanguages objectAtIndex: 0]; if (![lng length]) lng = defaultLanguage; return lng; } + (NSString *) fallbackIMAP4Server { return fallbackIMAP4Server; } + (SOGoUser *) userWithLogin: (NSString *) newLogin roles: (NSArray *) newRoles { SOGoCache *cache; SOGoUser *user; cache = [SOGoCache sharedCache]; user = [cache userNamed: newLogin]; if (!user) { user = [[self alloc] initWithLogin: newLogin roles: newRoles]; if (user) { [user autorelease]; [cache registerUser: user]; } } [user setPrimaryRoles: newRoles]; return user; } - (id) init { if ((self = [super init])) { userDefaults = nil; userSettings = nil; allEmails = nil; language = nil; currentPassword = nil; dateFormatter = nil; homeFolder = nil; } return self; } - (id) initWithLogin: (NSString *) newLogin roles: (NSArray *) newRoles { LDAPUserManager *um; NSString *realUID; if (acceptAnyUser || [newLogin isEqualToString: @"anonymous"] || [newLogin isEqualToString: @"freebusy"]) realUID = newLogin; else { um = [LDAPUserManager sharedUserManager]; realUID = [[um contactInfosForUserWithUIDorEmail: newLogin] objectForKey: @"c_uid"]; } if ([realUID length]) self = [super initWithLogin: realUID roles: newRoles]; else { [self release]; self = nil; } return self; } - (void) dealloc { [currentPassword release]; [userDefaults release]; [userSettings release]; [allEmails release]; [language release]; [dateFormatter release]; [homeFolder release]; [super dealloc]; } - (void) setPrimaryRoles: (NSArray *) newRoles { ASSIGN (roles, newRoles); } - (void) setCurrentPassword: (NSString *) newPassword { ASSIGN (currentPassword, newPassword); } - (NSString *) currentPassword { return currentPassword; } - (id) _fetchFieldForUser: (NSString *) field { NSDictionary *contactInfos; LDAPUserManager *um; um = [LDAPUserManager sharedUserManager]; contactInfos = [um contactInfosForUserWithUIDorEmail: login]; return [contactInfos objectForKey: field]; } - (void) _fetchAllEmails { allEmails = [self _fetchFieldForUser: @"emails"]; [allEmails retain]; } - (void) _fetchCN { cn = [self _fetchFieldForUser: @"cn"]; [cn retain]; } /* properties */ // - (NSString *) fullEmail // { // return [[LDAPUserManager sharedUserManager] getFullEmailForUID: login]; // } // - (NSString *) primaryEmail // { // if (!allEmails) // [self _fetchAllEmails]; // return [allEmails objectAtIndex: 0]; // } - (NSArray *) allEmails { if (!allEmails) [self _fetchAllEmails]; return allEmails; } - (NSString *) systemEmail { if (!allEmails) [self _fetchAllEmails]; return [allEmails lastObject]; } - (BOOL) hasEmail: (NSString *) email { if (!allEmails) [self _fetchAllEmails]; return [allEmails containsCaseInsensitiveString: email]; } - (NSString *) cn { if (!cn) [self _fetchCN]; return cn; } - (NSMutableDictionary *) defaultIdentity { NSMutableDictionary *currentIdentity, *defaultIdentity; NSEnumerator *identities; defaultIdentity = nil; identities = [[self allIdentities] objectEnumerator]; while (!defaultIdentity && (currentIdentity = [identities nextObject])) if ([[currentIdentity objectForKey: @"isDefault"] boolValue]) defaultIdentity = currentIdentity; return defaultIdentity; } - (void) saveMailAccounts { BOOL doSave; doSave = YES; if (!fallbackIsConfigured) { [self logWithFormat: @"'SOGoFallbackIMAP4Server' is not set"]; doSave = NO; } if (![LDAPUserManager defaultMailDomainIsConfigured]) { [self logWithFormat: @"'SOGoDefaultMailDomain' is not set"]; doSave = NO; } if (doSave) [userDefaults setObject: [self mailAccounts] forKey: @"MailAccounts"]; else [self logWithFormat: @"saving mail accounts is disabled until the" @" variable(s) mentionned above are configured"]; } // - (NSString *) primaryMailServer // { // return [[self userManager] getServerForUID: [self login]]; // } // - (NSArray *) additionalIMAP4AccountStrings // { // return [[self userManager]getSharedMailboxAccountStringsForUID: [self login]]; // } // - (NSArray *) additionalEMailAddresses // { // return [[self userManager] getSharedMailboxEMailsForUID: [self login]]; // } // - (NSDictionary *) additionalIMAP4AccountsAndEMails // { // return [[self userManager] getSharedMailboxesAndEMailsForUID: [self login]]; // } - (NSURL *) freeBusyURL { return nil; } - (SOGoDateFormatter *) dateFormatterInContext: (WOContext *) context { NSString *format; NSUserDefaults *ud; if (!dateFormatter) { dateFormatter = [SOGoDateFormatter new]; [dateFormatter setLocale: [context valueForKey: @"locale"]]; ud = [self userDefaults]; format = [ud stringForKey: @"ShortDateFormat"]; if (format) [dateFormatter setShortDateFormat: format]; format = [ud stringForKey: @"LongDateFormat"]; if (format) [dateFormatter setLongDateFormat: format]; format = [ud stringForKey: @"TimeFormat"]; if (format) [dateFormatter setTimeFormat: format]; } return dateFormatter; } /* defaults */ - (NSUserDefaults *) userDefaults { if (!userDefaults) userDefaults = [[AgenorUserDefaults alloc] initWithTableURL: AgenorProfileURL uid: login fieldName: @"c_defaults"]; return userDefaults; } - (NSUserDefaults *) userSettings { if (!userSettings) userSettings = [[AgenorUserDefaults alloc] initWithTableURL: AgenorProfileURL uid: login fieldName: @"c_settings"]; return userSettings; } - (NSString *) language { if (!language) { language = [[self userDefaults] stringForKey: @"Language"]; if (![language length]) language = [SOGoUser language]; [language retain]; } return language; } - (NSTimeZone *) timeZone { NSString *timeZoneName; if (!userTimeZone) { timeZoneName = [[self userDefaults] stringForKey: @"TimeZone"]; if ([timeZoneName length] > 0) userTimeZone = [NSTimeZone timeZoneWithName: timeZoneName]; if (!userTimeZone) userTimeZone = serverTimeZone; [userTimeZone retain]; } return userTimeZone; } - (NSTimeZone *) serverTimeZone { return serverTimeZone; } - (unsigned int) firstDayOfWeek { unsigned int firstDayOfWeek; NSNumber *value; value = [[self userDefaults] objectForKey: @"WeekStartDay"]; if (value) firstDayOfWeek = [value unsignedIntValue]; else firstDayOfWeek = sogoFirstDayOfWeek; return firstDayOfWeek; } - (NSCalendarDate *) firstDayOfWeekForDate: (NSCalendarDate *) date { int offset; NSCalendarDate *firstDay; offset = ([self firstDayOfWeek] - [date dayOfWeek]); if (offset > 0) offset -= 7; firstDay = [date addTimeInterval: offset * 86400]; return firstDay; } - (unsigned int) dayOfWeekForDate: (NSCalendarDate *) date { unsigned int offset, baseDayOfWeek, dayOfWeek; offset = [self firstDayOfWeek]; baseDayOfWeek = [date dayOfWeek]; if (offset > baseDayOfWeek) baseDayOfWeek += 7; dayOfWeek = baseDayOfWeek - offset; return dayOfWeek; } - (unsigned int) dayStartHour { int limit; limit = _timeValue ([[self userDefaults] stringForKey: @"DayStartTime"]); if (limit == -1) limit = defaultDayStartTime; return limit; } - (unsigned int) dayEndHour { int limit; limit = _timeValue ([[self userDefaults] stringForKey: @"DayEndTime"]); if (limit == -1) limit = defaultDayEndTime; return limit; } - (NSCalendarDate *) firstWeekOfYearForDate: (NSCalendarDate *) date { NSString *firstWeekRule; NSCalendarDate *januaryFirst, *firstWeek; unsigned int dayOfWeek; firstWeekRule = [userDefaults objectForKey: @"FirstWeek"]; januaryFirst = [NSCalendarDate dateWithYear: [date yearOfCommonEra] month: 1 day: 1 hour: 0 minute: 0 second: 0 timeZone: [date timeZone]]; if ([firstWeekRule isEqualToString: SOGoWeekStartFirst4DayWeek]) { dayOfWeek = [self dayOfWeekForDate: januaryFirst]; if (dayOfWeek < 4) firstWeek = [self firstDayOfWeekForDate: januaryFirst]; else firstWeek = [self firstDayOfWeekForDate: [januaryFirst dateByAddingYears: 0 months: 0 days: 7]]; } else if ([firstWeekRule isEqualToString: SOGoWeekStartFirstFullWeek]) { dayOfWeek = [self dayOfWeekForDate: januaryFirst]; if (dayOfWeek == 0) firstWeek = [self firstDayOfWeekForDate: januaryFirst]; else firstWeek = [self firstDayOfWeekForDate: [januaryFirst dateByAddingYears: 0 months: 0 days: 7]]; } else firstWeek = [self firstDayOfWeekForDate: januaryFirst]; return firstWeek; } - (unsigned int) weekNumberForDate: (NSCalendarDate *) date { NSCalendarDate *firstWeek; unsigned int weekNumber; firstWeek = [self firstWeekOfYearForDate: date]; if ([firstWeek earlierDate: date] == firstWeek) weekNumber = ([date timeIntervalSinceDate: firstWeek] / (86400 * 7) + 1); else weekNumber = 0; return weekNumber; } /* mail */ - (void) _prepareDefaultMailAccounts { NSMutableDictionary *mailAccount, *identity; NSMutableArray *identities; NSString *name, *fullName, *imapLogin; NSArray *mails; unsigned int count, max; imapLogin = [[LDAPUserManager sharedUserManager] getImapLoginForUID: login]; mailAccount = [NSMutableDictionary dictionary]; name = [NSString stringWithFormat: @"%@@%@", imapLogin, fallbackIMAP4Server]; [mailAccount setObject: imapLogin forKey: @"userName"]; [mailAccount setObject: fallbackIMAP4Server forKey: @"serverName"]; [mailAccount setObject: name forKey: @"name"]; identities = [NSMutableArray array]; mails = [self allEmails]; max = [mails count]; if (max > 1) max--; for (count = 0; count < max; count++) { identity = [NSMutableDictionary dictionary]; fullName = [self cn]; if (![fullName length]) fullName = login; [identity setObject: fullName forKey: @"fullName"]; [identity setObject: [mails objectAtIndex: count] forKey: @"email"]; [identities addObject: identity]; } [[identities objectAtIndex: 0] setObject: [NSNumber numberWithBool: YES] forKey: @"isDefault"]; [mailAccount setObject: identities forKey: @"identities"]; mailAccounts = [NSMutableArray new]; [mailAccounts addObject: mailAccount]; } - (NSArray *) mailAccounts { NSUserDefaults *ud; if (!mailAccounts) { ud = [self userDefaults]; mailAccounts = [ud objectForKey: @"MailAccounts"]; if (mailAccounts) [mailAccounts retain]; else [self _prepareDefaultMailAccounts]; } return mailAccounts; } - (NSDictionary *) accountWithName: (NSString *) accountName; { NSEnumerator *accounts; NSDictionary *mailAccount, *currentAccount; mailAccount = nil; accounts = [[self mailAccounts] objectEnumerator]; while (!mailAccount && ((currentAccount = [accounts nextObject]))) if ([[currentAccount objectForKey: @"name"] isEqualToString: accountName]) mailAccount = currentAccount; return mailAccount; } /* @interface SOGoMailIdentity : NSObject { NSString *name; NSString *email; NSString *replyTo; NSString *organization; NSString *signature; NSString *vCard; NSString *sentFolderName; NSString *sentBCC; NSString *draftsFolderName; NSString *templatesFolderName; struct { int composeHTML:1; int reserved:31; } idFlags; } - (void) setName: (NSString *) _value; - (NSString *) name; - (void) setEmail: (NSString *) _value; - (NSString *) email; - (void) setReplyTo: (NSString *) _value; - (NSString *) replyTo; - (void) setOrganization: (NSString *) _value; - (NSString *) organization; - (void) setSignature: (NSString *) _value; - (NSString *) signature; - (BOOL) hasSignature; - (void) setVCard: (NSString *) _value; - (NSString *) vCard; - (BOOL) hasVCard; - (void) setSentFolderName: (NSString *) _value; - (NSString *) sentFolderName; - (void) setSentBCC: (NSString *) _value; - (NSString *) sentBCC; - (void) setDraftsFolderName: (NSString *) _value; - (NSString *) draftsFolderName; - (void) setTemplatesFolderName: (NSString *) _value; - (NSString *) templatesFolderName; @end */ - (NSArray *) allIdentities { NSArray *identities; [self mailAccounts]; identities = [mailAccounts objectsForKey: @"identities"]; return [identities flattenedArray]; } - (NSDictionary *) primaryIdentity { NSDictionary *defaultAccount; [self mailAccounts]; defaultAccount = [mailAccounts objectAtIndex: 0]; return [[defaultAccount objectForKey: @"identities"] objectAtIndex: 0]; } - (NSString *) signature { return [[self primaryIdentity] objectForKey: @"signature"]; } - (NSString *) messageForwarding { NSString *messageForwarding; messageForwarding = [[self userDefaults] stringForKey: @"MessageForwarding"]; if (![messageForwarding length]) messageForwarding = @"inline"; return messageForwarding; } - (NSString *) messageCheck { NSString *messageCheck; messageCheck = [[self userDefaults] stringForKey: @"MessageCheck"]; if (![messageCheck length]) messageCheck = @"manually"; return messageCheck; } /* folders */ // TODO: those methods should check whether the traversal stack in the context // already contains proper folders to improve caching behaviour - (SOGoUserFolder *) homeFolderInContext: (id) context { if (!homeFolder) { homeFolder = [[WOApplication application] lookupName: [self login] inContext: context acquire: NO]; [homeFolder retain]; } return homeFolder; } - (SOGoAppointmentFolders *) calendarsFolderInContext: (WOContext *) context { return [[self homeFolderInContext: context] lookupName: @"Calendar" inContext: context acquire: NO]; } - (SOGoAppointmentFolder *) personalCalendarFolderInContext: (WOContext *) context { return [[self calendarsFolderInContext: context] lookupName: @"personal" inContext: context acquire: NO]; } // - (id) schedulingCalendarInContext: (id) _ctx // { // /* Note: watch out for cyclic references */ // id folder; // folder = [(WOContext *)_ctx objectForKey:@"ActiveUserCalendar"]; // if (folder != nil) // return [folder isNotNull] ? folder : nil; // folder = [self homeFolderInContext:_ctx]; // if ([folder isKindOfClass:[NSException class]]) // return folder; // folder = [folder lookupName:@"Calendar" inContext:_ctx acquire:NO]; // if ([folder isKindOfClass:[NSException class]]) // return folder; // [(WOContext *)_ctx setObject:folder ? folder : [NSNull null] // forKey:@"ActiveUserCalendar"]; // return folder; // } - (NSArray *) rolesForObject: (NSObject *) object inContext: (WOContext *) context { NSMutableArray *rolesForObject; NSArray *sogoRoles; NSString *rqMethod; rolesForObject = [NSMutableArray array]; sogoRoles = [super rolesForObject: object inContext: context]; if (sogoRoles) [rolesForObject addObjectsFromArray: sogoRoles]; if ((superUsernames && [superUsernames containsObject: login]) || [[object ownerInContext: context] isEqualToString: login]) [rolesForObject addObject: SoRole_Owner]; if ([object isKindOfClass: [SOGoObject class]]) { sogoRoles = [(SOGoObject *) object aclsForUser: login]; if ([sogoRoles count]) [rolesForObject addObjectsFromArray: sogoRoles]; sogoRoles = [(SOGoObject *) object subscriptionRoles]; if ([sogoRoles firstObjectCommonWithArray: rolesForObject]) [rolesForObject addObject: SOGoRole_AuthorizedSubscriber]; } #warning this is a hack to work-around the poor implementation of PROPPATCH in SOPE rqMethod = [[context request] method]; if ([rqMethod isEqualToString: @"PROPPATCH"]) [rolesForObject addObject: @"PROPPATCHer"]; return rolesForObject; } - (BOOL) isEqual: (id) otherUser { return ([otherUser isKindOfClass: [SoUser class]] && [login isEqualToString: [otherUser login]]); } - (BOOL) isSuperUser { return [superUsernames containsObject: login]; } /* module access */ - (BOOL) canAccessModule: (NSString *) module { NSString *accessValue; accessValue = [self _fetchFieldForUser: [NSString stringWithFormat: @"%@Access", module]]; return [accessValue boolValue]; } @end /* SOGoUser */