/* MAPIStoreMapping.m - this file is part of SOGo * * Copyright (C) 2010-2012 Inverse inc. * * 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 3, 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. */ #include #import #import #import #import #import #import #import #import #undef DEBUG #include #include #import "MAPIStoreTypes.h" #import "MAPIStoreMapping.h" #include static NSMutableDictionary *mappingRegistry = nil; @implementation MAPIStoreMapping + (void) initialize { mappingRegistry = [NSMutableDictionary new]; } static inline id MAPIStoreMappingKeyFromId (uint64_t idNbr) { return [NSString stringWithUnsignedLongLong: idNbr]; } + (id) mappingForUsername: (NSString *) username withIndexing: (struct indexing_context *) indexing { id mapping; mapping = [mappingRegistry objectForKey: username]; if (!mapping) { mapping = [[self alloc] initForUsername: username withIndexing: indexing]; [mapping autorelease]; } return mapping; } - (id) init { if ((self = [super init])) { memCtx = talloc_zero (NULL, TALLOC_CTX); indexing = NULL; useCount = 0; } return self; } - (void) increaseUseCount { if (useCount == 0) { [mappingRegistry setObject: self forKey: username]; [self logWithFormat: @"mapping registered (%@)", username]; } useCount++; } - (void) decreaseUseCount { useCount--; if (useCount == 0) { [mappingRegistry removeObjectForKey: username]; [self logWithFormat: @"mapping deregistered (%@)", username]; } } - (id) initForUsername: (NSString *) newUsername withIndexing: (struct indexing_context *) newIndexing { if ((self = [self init])) { ASSIGN (username, newUsername); indexing = newIndexing; /* Workaround so all indexing context are valid and won't be freed. */ // TODO refactor indexing interface talloc_reference(memCtx, indexing); } return self; } - (void) dealloc { [username release]; talloc_free (memCtx); [super dealloc]; } - (NSString *) urlFromID: (uint64_t) idNbr { char* url = NULL; enum mapistore_error ret; bool soft_delete = false; ret = indexing->get_uri(indexing, [username UTF8String], memCtx, idNbr, &url, &soft_delete); if (ret != MAPISTORE_SUCCESS) return NULL; NSString *res = [[[NSString alloc] initWithUTF8String:url] autorelease]; talloc_free(url); return res; } - (uint64_t) idFromURL: (NSString *) url { enum mapistore_error ret; uint64_t idNbr; bool softDeleted; ret = indexing->get_fmid(indexing, [username UTF8String], [url UTF8String], false, &idNbr, &softDeleted); if (ret == MAPISTORE_SUCCESS && !softDeleted) return idNbr; else return NSNotFound; } - (uint64_t) idFromURL: (NSString *) url isSoftDeleted: (bool *) softDeleted { enum mapistore_error ret; uint64_t idNbr; ret = indexing->get_fmid(indexing, [username UTF8String], [url UTF8String], false, &idNbr, softDeleted); if (ret != MAPISTORE_SUCCESS) return NSNotFound; return idNbr; } - (void) _updateFolderWithURL: (NSString *) oldURL withURL: (NSString *) urlString { const char *searchURL; uint64_t idNbr; bool softDeleted; NSString *current; NSString *newURL; if ([oldURL isEqualToString: urlString]) return; searchURL = [[oldURL stringByAppendingString:@"*"] UTF8String]; while (indexing->get_fmid(indexing, [username UTF8String], searchURL,true, &idNbr, &softDeleted) == MAPISTORE_SUCCESS) { // Ignore deleted if (softDeleted) continue; current = [self urlFromID:idNbr]; newURL = [current stringByReplacingPrefix: oldURL withPrefix: urlString]; indexing->update_fmid(indexing, [username UTF8String], idNbr, [newURL UTF8String]); } } - (void) updateID: (uint64_t) idNbr withURL: (NSString *) urlString { NSString *oldURL; oldURL = [self urlFromID: idNbr]; if (oldURL) { if ([oldURL hasSuffix: @"/"]) /* is container ? */ { if (![urlString hasSuffix: @"/"]) [NSException raise: NSInvalidArgumentException format: @"a container url must have an ending '/'"]; [self _updateFolderWithURL: oldURL withURL: urlString]; } else { if ([urlString hasSuffix: @"/"]) [NSException raise: NSInvalidArgumentException format: @"a leaf url must not have an ending '/'"]; indexing->update_fmid(indexing, [username UTF8String], idNbr, [urlString UTF8String]); } } } - (BOOL) updateURL: (NSString *) urlString withID: (uint64_t) idNbr { BOOL rc = NO; uint64_t oldIDNbr; oldIDNbr = [self idFromURL: urlString]; if (oldIDNbr) { [self logWithFormat: @"Updating URL: %@ with id %.16"PRIx64" from old id %.16"PRIx64, urlString, idNbr, oldIDNbr]; [self unregisterURLWithID: oldIDNbr]; [self registerURL: urlString withID: idNbr]; rc = YES; } return rc; } - (BOOL) registerURL: (NSString *) urlString withID: (uint64_t) idNbr { NSString *oldURL; uint64_t oldIdNbr; bool softDeleted; oldURL = [self urlFromID: idNbr]; if (oldURL != NULL) { [self errorWithFormat: @"url with idNbr already registered: (oldUrl='%@', newUrl='%@', id=x%.16"PRIx64")", oldURL, urlString, idNbr]; return NO; } oldIdNbr = [self idFromURL: urlString isSoftDeleted: &softDeleted]; if (oldIdNbr != NSNotFound) { if (softDeleted) [self logWithFormat: @"Attempt to register a soft-deleted %@", urlString]; else [self errorWithFormat: @"attempt to double register an entry with idNbr ('%@', %lld," @" 0x%.16"PRIx64", oldid=0x%.16"PRIx64")", urlString, idNbr, idNbr, oldIdNbr]; return NO; } else { // [self logWithFormat: @"registered url '%@' with id %lld (0x%.16"PRIx64")", // urlString, idNbr, idNbr]; /* Add the record given its fid and mapistore_uri */ indexing->add_fmid(indexing, [username UTF8String], idNbr, [urlString UTF8String]); } return YES; } - (void) registerURLs: (NSArray *) urlStrings withIDs: (NSArray *) idNbrs { uint64_t count, max, newID; max = [urlStrings count]; if (max == [idNbrs count]) { for (count = 0; count < max; count++) { newID = [[idNbrs objectAtIndex: count] unsignedLongLongValue]; [self registerURL: [urlStrings objectAtIndex: count] withID: newID]; } } else [NSException raise: NSInvalidArgumentException format: @"number of urls and ids do not match"]; } - (void) unregisterURLWithID: (uint64_t) idNbr { indexing->del_fmid(indexing, [username UTF8String], idNbr, MAPISTORE_PERMANENT_DELETE); } - (void) unregisterURLWithID: (uint64_t) idNbr andFlags: (uint8_t) flags { indexing->del_fmid(indexing, [username UTF8String], idNbr, (flags == MAPISTORE_SOFT_DELETE) ? MAPISTORE_SOFT_DELETE : MAPISTORE_PERMANENT_DELETE); } @end