72592597ca
Monotone-Revision: b1fef274eded93af1afedb293db6a9921fadf828 Monotone-Author: flachapelle@inverse.ca Monotone-Date: 2009-02-17T14:30:49 Monotone-Branch: ca.inverse.sogo
409 lines
12 KiB
Objective-C
409 lines
12 KiB
Objective-C
/*
|
|
Copyright (C) 2006-2009 Inverse inc.
|
|
Copyright (C) 2004-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 <Foundation/NSArray.h>
|
|
#import <Foundation/NSEnumerator.h>
|
|
#import <Foundation/NSString.h>
|
|
|
|
#import <NGObjWeb/NSException+HTTP.h>
|
|
#import <NGObjWeb/SoObject+SoDAV.h>
|
|
#import <NGObjWeb/WOContext.h>
|
|
#import <NGObjWeb/WORequest.h>
|
|
#import <NGObjWeb/WOResponse.h>
|
|
#import <NGExtensions/NSObject+Logs.h>
|
|
#import <NGExtensions/NSString+misc.h>
|
|
#import <NGCards/CardGroup.h>
|
|
#import <EOControl/EOQualifier.h>
|
|
#import <EOControl/EOSortOrdering.h>
|
|
#import <GDLContentStore/GCSFolder.h>
|
|
|
|
#import <SOGo/SOGoCache.h>
|
|
#import <SOGo/NSArray+Utilities.h>
|
|
#import <SOGo/NSDictionary+Utilities.h>
|
|
#import <SOGo/NSString+Utilities.h>
|
|
|
|
#import "SOGoContactGCSEntry.h"
|
|
#import "SOGoContactGCSList.h"
|
|
|
|
#import "SOGoContactGCSFolder.h"
|
|
|
|
#define folderListingFields [NSArray arrayWithObjects: @"c_name", @"c_cn", \
|
|
@"c_givenname", @"c_sn", @"c_screenname", \
|
|
@"c_o", @"c_mail", @"c_telephonenumber", \
|
|
nil]
|
|
|
|
@implementation SOGoContactGCSFolder
|
|
|
|
/* name lookup */
|
|
|
|
//- (id <SOGoContactObject>) lookupContactWithId: (NSString *) recordId
|
|
//{
|
|
// SOGoContactGCSEntry *contact;
|
|
//
|
|
// if ([recordId length] > 0)
|
|
// contact = [SOGoContactGCSEntry objectWithName: recordId
|
|
// inContainer: self];
|
|
// else
|
|
// contact = nil;
|
|
//
|
|
// return contact;
|
|
//}
|
|
|
|
- (Class) objectClassForContent: (NSString *) content
|
|
{
|
|
CardGroup *cardEntry;
|
|
NSString *firstTag;
|
|
Class objectClass;
|
|
|
|
objectClass = Nil;
|
|
|
|
cardEntry = [CardGroup parseSingleFromSource: content];
|
|
if (cardEntry)
|
|
{
|
|
firstTag = [[cardEntry tag] uppercaseString];
|
|
if ([firstTag isEqualToString: @"VCARD"])
|
|
objectClass = [SOGoContactGCSEntry class];
|
|
else if ([firstTag isEqualToString: @"VLIST"])
|
|
objectClass = [SOGoContactGCSList class];
|
|
}
|
|
|
|
return objectClass;
|
|
}
|
|
|
|
- (Class) objectClassForComponentName: (NSString *) componentName
|
|
{
|
|
Class objectClass;
|
|
|
|
if ([componentName isEqualToString: @"vcard"])
|
|
objectClass = [SOGoContactGCSEntry class];
|
|
else if ([componentName isEqualToString: @"vlist"])
|
|
objectClass = [SOGoContactGCSList class];
|
|
else
|
|
objectClass = Nil;
|
|
|
|
return objectClass;
|
|
}
|
|
|
|
- (Class) objectClassForResourceNamed: (NSString *) name
|
|
{
|
|
EOQualifier *qualifier;
|
|
NSArray *records;
|
|
NSString *component;
|
|
Class objectClass;
|
|
|
|
qualifier = [EOQualifier qualifierWithQualifierFormat:@"c_name = %@", name];
|
|
records = [[self ocsFolder] fetchFields: [NSArray arrayWithObject: @"c_component"]
|
|
matchingQualifier: qualifier];
|
|
|
|
if ([records count])
|
|
{
|
|
component = [[records objectAtIndex: 0] valueForKey: @"c_component"];
|
|
objectClass = [self objectClassForComponentName: component];
|
|
}
|
|
else
|
|
objectClass = Nil;
|
|
|
|
return objectClass;
|
|
}
|
|
|
|
- (BOOL) requestNamedIsHandledLater: (NSString *) name
|
|
{
|
|
return [name isEqualToString: @"OPTIONS"];
|
|
}
|
|
|
|
- (id) lookupName: (NSString *)_key
|
|
inContext: (id)_ctx
|
|
acquire: (BOOL)_flag
|
|
{
|
|
id obj;
|
|
NSString *url;
|
|
BOOL handledLater;
|
|
|
|
/* first check attributes directly bound to the application */
|
|
handledLater = [self requestNamedIsHandledLater: _key];
|
|
if (handledLater)
|
|
obj = nil;
|
|
else
|
|
{
|
|
obj = [super lookupName:_key inContext:_ctx acquire:NO];
|
|
if (!obj)
|
|
{
|
|
if ([self isValidContentName: _key])
|
|
{
|
|
url = [[[_ctx request] uri] urlWithoutParameters];
|
|
if ([url hasSuffix: @"AsContact"])
|
|
obj = [SOGoContactGCSEntry objectWithName: _key
|
|
inContainer: self];
|
|
else if ([url hasSuffix: @"AsList"])
|
|
obj = [SOGoContactGCSList objectWithName: _key
|
|
inContainer: self];
|
|
[obj setIsNew: YES];
|
|
}
|
|
}
|
|
if (!obj)
|
|
obj = [NSException exceptionWithHTTPStatus:404 /* Not Found */];
|
|
}
|
|
|
|
if (obj)
|
|
[[SOGoCache sharedCache] registerObject: obj
|
|
withName: _key
|
|
inContainer: container];
|
|
|
|
return obj;
|
|
}
|
|
|
|
// - (id) lookupName: (NSString *) _key
|
|
// inContext: (WOContext *) _ctx
|
|
// acquire: (BOOL) _flag
|
|
// {
|
|
// id obj;
|
|
// BOOL isPut;
|
|
|
|
// isPut = NO;
|
|
// obj = [super lookupName:_key inContext:_ctx acquire:NO];
|
|
|
|
// if (!obj)
|
|
// {
|
|
// if ([[[_ctx request] method] isEqualToString: @"PUT"])
|
|
// {
|
|
// if ([_key isEqualToString: @"PUT"])
|
|
// isPut = YES;
|
|
// else
|
|
|
|
// obj = [SOGoContactGCSEntry objectWithName: _key
|
|
// inContainer: self];
|
|
// }
|
|
// else
|
|
// obj = [self lookupContactWithId: _key];
|
|
// }
|
|
// if (!(obj || isPut))
|
|
// obj = [NSException exceptionWithHTTPStatus:404 /* Not Found */];
|
|
|
|
// #if 0
|
|
// if ([[self ocsFolder] versionOfContentWithName:_key])
|
|
// #endif
|
|
// return [self contactWithName:_key inContext:_ctx];
|
|
// }
|
|
|
|
// /* return 404 to stop acquisition */
|
|
// return obj;
|
|
// }
|
|
|
|
/* fetching */
|
|
- (EOQualifier *) _qualifierForFilter: (NSString *) filter
|
|
{
|
|
NSString *qs;
|
|
EOQualifier *qualifier;
|
|
|
|
if (filter && [filter length] > 0)
|
|
{
|
|
filter = [filter stringByReplacingString: @"'" withString: @"\\'\\'"];
|
|
qs = [NSString stringWithFormat:
|
|
@"(c_sn isCaseInsensitiveLike: '%@%%') OR "
|
|
@"(c_givenname isCaseInsensitiveLike: '%@%%') OR "
|
|
@"(c_cn isCaseInsensitiveLike: '%@%%') OR "
|
|
@"(c_mail isCaseInsensitiveLike: '%@%%') OR "
|
|
@"(c_telephonenumber isCaseInsensitiveLike: '%%%@%%')",
|
|
filter, filter, filter, filter, filter];
|
|
qualifier = [EOQualifier qualifierWithQualifierFormat: qs];
|
|
}
|
|
else
|
|
qualifier = nil;
|
|
|
|
return qualifier;
|
|
}
|
|
|
|
- (void) fixupContactRecord: (NSMutableDictionary *) contactRecord
|
|
{
|
|
NSString *data;
|
|
|
|
data = [contactRecord objectForKey: @"c_cn"];
|
|
if (![data length])
|
|
{
|
|
data = [contactRecord keysWithFormat: @"%{c_givenname} %{c_sn}"];
|
|
[contactRecord setObject: data forKey: @"c_cn"];
|
|
}
|
|
|
|
if (![contactRecord objectForKey: @"c_mail"])
|
|
[contactRecord setObject: @"" forKey: @"c_mail"];
|
|
if (![contactRecord objectForKey: @"c_screenname"])
|
|
[contactRecord setObject: @"" forKey: @"c_screenname"];
|
|
if (![contactRecord objectForKey: @"c_o"])
|
|
[contactRecord setObject: @"" forKey: @"c_o"];
|
|
if (![contactRecord objectForKey: @"c_telephonenumber"])
|
|
[contactRecord setObject: @"" forKey: @"c_telephonenumber"];
|
|
}
|
|
|
|
- (NSArray *) _flattenedRecords: (NSArray *) records
|
|
{
|
|
NSMutableArray *newRecords;
|
|
NSEnumerator *oldRecords;
|
|
NSDictionary *oldRecord;
|
|
NSMutableDictionary *newRecord;
|
|
|
|
newRecords = [NSMutableArray arrayWithCapacity: [records count]];
|
|
|
|
oldRecords = [records objectEnumerator];
|
|
oldRecord = [oldRecords nextObject];
|
|
while (oldRecord)
|
|
{
|
|
newRecord = [NSMutableDictionary dictionaryWithDictionary: oldRecord];
|
|
[self fixupContactRecord: newRecord];
|
|
[newRecords addObject: newRecord];
|
|
oldRecord = [oldRecords nextObject];
|
|
}
|
|
|
|
return newRecords;
|
|
}
|
|
|
|
- (NSArray *) lookupContactsWithFilter: (NSString *) filter
|
|
sortBy: (NSString *) sortKey
|
|
ordering: (NSComparisonResult) sortOrdering
|
|
{
|
|
NSArray *fields, *dbRecords, *records;
|
|
EOQualifier *qualifier;
|
|
EOSortOrdering *ordering;
|
|
|
|
fields = folderListingFields;
|
|
qualifier = [self _qualifierForFilter: filter];
|
|
dbRecords = [[self ocsFolder] fetchFields: fields
|
|
matchingQualifier: qualifier];
|
|
|
|
if ([dbRecords count] > 0)
|
|
{
|
|
records = [self _flattenedRecords: dbRecords];
|
|
ordering
|
|
= [EOSortOrdering sortOrderingWithKey: sortKey
|
|
selector: ((sortOrdering == NSOrderedDescending)
|
|
? EOCompareCaseInsensitiveDescending
|
|
: EOCompareCaseInsensitiveAscending)];
|
|
records
|
|
= [records sortedArrayUsingKeyOrderArray:
|
|
[NSArray arrayWithObject: ordering]];
|
|
}
|
|
else
|
|
records = nil;
|
|
// else
|
|
//[self errorWithFormat:@"(%s): fetch failed!", __PRETTY_FUNCTION__];
|
|
|
|
[self debugWithFormat:@"fetched %i records.", [records count]];
|
|
return records;
|
|
}
|
|
|
|
#warning this should be unified within SOGoFolder
|
|
- (void) appendObject: (NSDictionary *) object
|
|
withBaseURL: (NSString *) baseURL
|
|
toREPORTResponse: (WOResponse *) r
|
|
{
|
|
SOGoContactGCSEntry *component;
|
|
NSString *name, *etagLine, *contactString;
|
|
|
|
name = [object objectForKey: @"c_name"];
|
|
component = [self lookupName: name inContext: context acquire: NO];
|
|
|
|
[r appendContentString: @" <D:response>\r\n"];
|
|
[r appendContentString: @" <D:href>"];
|
|
[r appendContentString: baseURL];
|
|
if (![baseURL hasSuffix: @"/"])
|
|
[r appendContentString: @"/"];
|
|
[r appendContentString: name];
|
|
[r appendContentString: @"</D:href>\r\n"];
|
|
|
|
[r appendContentString: @" <D:propstat>\r\n"];
|
|
[r appendContentString: @" <D:prop>\r\n"];
|
|
etagLine = [NSString stringWithFormat: @" <D:getetag>%@</D:getetag>\r\n",
|
|
[component davEntityTag]];
|
|
[r appendContentString: etagLine];
|
|
[r appendContentString: @" </D:prop>\r\n"];
|
|
[r appendContentString: @" <D:status>HTTP/1.1 200 OK</D:status>\r\n"];
|
|
[r appendContentString: @" </D:propstat>\r\n"];
|
|
[r appendContentString: @" <C:addressbook-data>"];
|
|
contactString = [[component contentAsString] stringByEscapingXMLString];
|
|
[r appendContentString: contactString];
|
|
[r appendContentString: @"</C:addressbook-data>\r\n"];
|
|
[r appendContentString: @" </D:response>\r\n"];
|
|
}
|
|
|
|
- (NSArray *) davComplianceClassesInContext: (id)_ctx
|
|
{
|
|
NSMutableArray *classes;
|
|
NSArray *primaryClasses;
|
|
|
|
classes = [NSMutableArray new];
|
|
[classes autorelease];
|
|
|
|
primaryClasses = [super davComplianceClassesInContext: _ctx];
|
|
if (primaryClasses)
|
|
[classes addObjectsFromArray: primaryClasses];
|
|
[classes addObject: @"access-control"];
|
|
[classes addObject: @"addressbook-access"];
|
|
|
|
return classes;
|
|
}
|
|
|
|
- (NSString *) groupDavResourceType
|
|
{
|
|
return @"vcard-collection";
|
|
}
|
|
|
|
- (NSArray *) davResourceType
|
|
{
|
|
NSMutableArray *resourceType;
|
|
NSArray *cardDavCollection;
|
|
|
|
cardDavCollection
|
|
= [NSArray arrayWithObjects: @"addressbook",
|
|
@"urn:ietf:params:xml:ns:carddav", nil];
|
|
resourceType = [NSMutableArray arrayWithArray: [super davResourceType]];
|
|
[resourceType addObject: cardDavCollection];
|
|
|
|
return resourceType;
|
|
}
|
|
|
|
/* sorting */
|
|
- (NSComparisonResult) compare: (id) otherFolder
|
|
{
|
|
NSComparisonResult comparison;
|
|
|
|
if ([NSStringFromClass([otherFolder class])
|
|
isEqualToString: @"SOGoContactLDAPFolder"])
|
|
comparison = NSOrderedAscending;
|
|
else
|
|
comparison = [super compare: otherFolder];
|
|
|
|
return comparison;
|
|
}
|
|
|
|
/* folder type */
|
|
|
|
- (NSString *) folderType
|
|
{
|
|
return @"Contact";
|
|
}
|
|
|
|
- (NSString *) outlookFolderClass
|
|
{
|
|
return @"IPF.Contact";
|
|
}
|
|
|
|
@end /* SOGoContactGCSFolder */
|