2010-10-01 22:35:29 +02:00
|
|
|
/* MAPIStoreMapping.m - this file is part of SOGo
|
2010-10-01 20:54:30 +02:00
|
|
|
*
|
2012-08-17 21:04:57 +02:00
|
|
|
* Copyright (C) 2010-2012 Inverse inc.
|
2010-10-01 20:54:30 +02:00
|
|
|
*
|
|
|
|
* Author: Wolfgang Sourdeau <wsourdeau@inverse.ca>
|
|
|
|
*
|
|
|
|
* 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
|
2010-10-18 14:57:31 +02:00
|
|
|
* the Free Software Foundation; either version 3, or (at your option)
|
2010-10-01 20:54:30 +02:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2012-10-10 17:55:06 +02:00
|
|
|
#include <inttypes.h>
|
|
|
|
|
2010-12-13 17:54:32 +01:00
|
|
|
#import <Foundation/NSArray.h>
|
2010-10-01 20:54:30 +02:00
|
|
|
#import <Foundation/NSDictionary.h>
|
2010-10-13 17:30:01 +02:00
|
|
|
#import <Foundation/NSException.h>
|
2010-10-01 20:54:30 +02:00
|
|
|
#import <Foundation/NSString.h>
|
|
|
|
#import <Foundation/NSValue.h>
|
|
|
|
|
|
|
|
#import <NGExtensions/NSObject+Logs.h>
|
2012-10-09 20:14:58 +02:00
|
|
|
#import <NGExtensions/NSObject+Values.h>
|
2010-10-01 20:54:30 +02:00
|
|
|
|
2012-08-13 19:48:57 +02:00
|
|
|
#import <SOGo/NSString+Utilities.h>
|
|
|
|
|
2011-06-04 01:53:30 +02:00
|
|
|
#import "MAPIStoreTypes.h"
|
|
|
|
|
2010-10-01 20:54:30 +02:00
|
|
|
#import "MAPIStoreMapping.h"
|
|
|
|
|
2010-12-13 17:54:32 +01:00
|
|
|
#include <talloc.h>
|
2011-06-04 01:53:30 +02:00
|
|
|
#include <tdb.h>
|
2011-10-07 12:27:59 +02:00
|
|
|
|
|
|
|
struct tdb_wrap {
|
|
|
|
struct tdb_context *tdb;
|
|
|
|
};
|
2010-12-13 17:54:32 +01:00
|
|
|
|
2011-08-10 20:32:53 +02:00
|
|
|
static NSMutableDictionary *mappingRegistry = nil;
|
|
|
|
|
2011-06-04 01:53:30 +02:00
|
|
|
@implementation MAPIStoreMapping
|
2010-12-13 17:54:32 +01:00
|
|
|
|
2011-08-10 20:32:53 +02:00
|
|
|
+ (void) initialize
|
|
|
|
{
|
|
|
|
mappingRegistry = [NSMutableDictionary new];
|
|
|
|
}
|
|
|
|
|
2012-10-03 17:25:47 +02:00
|
|
|
static inline id
|
|
|
|
MAPIStoreMappingKeyFromId (uint64_t idNbr)
|
|
|
|
{
|
2012-10-09 20:14:58 +02:00
|
|
|
return [NSString stringWithUnsignedLongLong: idNbr];
|
2012-10-03 17:25:47 +02:00
|
|
|
}
|
|
|
|
|
2010-12-13 17:54:32 +01:00
|
|
|
static int
|
|
|
|
MAPIStoreMappingTDBTraverse (TDB_CONTEXT *ctx, TDB_DATA data1, TDB_DATA data2,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
NSMutableDictionary *mapping;
|
2012-10-03 17:25:47 +02:00
|
|
|
id idKey;
|
2010-12-13 17:54:32 +01:00
|
|
|
NSString *uri;
|
|
|
|
char *idStr, *uriStr;
|
2012-10-03 17:25:47 +02:00
|
|
|
uint64_t idNbr;
|
2010-12-13 17:54:32 +01:00
|
|
|
|
2011-09-04 18:25:32 +02:00
|
|
|
// get the key
|
|
|
|
// key examples : key(18) = "0x6900000000000001"
|
|
|
|
// key(31) = "SOFT_DELETED:0xb100020000000001"
|
|
|
|
//
|
2010-12-13 17:54:32 +01:00
|
|
|
idStr = (char *) data1.dptr;
|
2012-10-03 17:25:47 +02:00
|
|
|
idKey = nil;
|
2011-09-04 18:25:32 +02:00
|
|
|
|
|
|
|
if (strncmp(idStr, "SOFT_DELETED:", 13) != 0)
|
|
|
|
{
|
|
|
|
// It's very important here to use strtoull and NOT strtoll as
|
|
|
|
// the latter will overflow a long long with typical key values.
|
2012-10-03 17:25:47 +02:00
|
|
|
idNbr = strtoull(idStr, NULL, 0);
|
|
|
|
// idKey = [NSNumber numberWithUnsignedLongLong: idNbr];
|
|
|
|
idKey = MAPIStoreMappingKeyFromId(idNbr);
|
2011-09-04 18:25:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// get the value and null-terminate it
|
|
|
|
uriStr = (char *)malloc(sizeof(char *) * data2.dsize+1);
|
|
|
|
memset(uriStr, 0, data2.dsize+1);
|
|
|
|
memcpy(uriStr, (const char *) data2.dptr, data2.dsize);
|
2010-12-13 17:54:32 +01:00
|
|
|
uri = [NSString stringWithUTF8String: uriStr];
|
|
|
|
free (uriStr);
|
|
|
|
|
|
|
|
mapping = data;
|
|
|
|
|
2012-10-03 17:25:47 +02:00
|
|
|
if (uri && idKey)
|
2011-09-04 18:25:32 +02:00
|
|
|
{
|
2012-10-03 17:25:47 +02:00
|
|
|
[mapping setObject: uri forKey: idKey];
|
2011-09-04 18:25:32 +02:00
|
|
|
}
|
2011-07-29 04:13:39 +02:00
|
|
|
|
2010-12-13 17:54:32 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-08-10 20:32:53 +02:00
|
|
|
+ (id) mappingForUsername: (NSString *) username
|
|
|
|
withIndexing: (struct tdb_wrap *) indexing
|
2010-12-13 17:54:32 +01:00
|
|
|
{
|
2011-08-10 20:32:53 +02:00
|
|
|
id mapping;
|
2010-12-13 17:54:32 +01:00
|
|
|
|
2011-08-10 20:32:53 +02:00
|
|
|
mapping = [mappingRegistry objectForKey: username];
|
|
|
|
if (!mapping)
|
|
|
|
{
|
|
|
|
mapping = [[self alloc] initForUsername: username
|
|
|
|
withIndexing: indexing];
|
|
|
|
[mapping autorelease];
|
|
|
|
}
|
2010-10-01 20:54:30 +02:00
|
|
|
|
2011-08-10 20:32:53 +02:00
|
|
|
return mapping;
|
2011-06-04 01:53:30 +02:00
|
|
|
}
|
2010-10-01 20:54:30 +02:00
|
|
|
|
2011-06-04 01:53:30 +02:00
|
|
|
- (id) init
|
2010-10-15 19:18:34 +02:00
|
|
|
{
|
2011-06-04 01:53:30 +02:00
|
|
|
if ((self = [super init]))
|
|
|
|
{
|
2011-10-17 17:29:14 +02:00
|
|
|
memCtx = talloc_zero (NULL, TALLOC_CTX);
|
2011-06-04 01:53:30 +02:00
|
|
|
mapping = [NSMutableDictionary new];
|
|
|
|
reverseMapping = [NSMutableDictionary new];
|
|
|
|
indexing = NULL;
|
2011-08-10 20:32:53 +02:00
|
|
|
useCount = 0;
|
2011-06-04 01:53:30 +02:00
|
|
|
}
|
2010-10-15 19:18:34 +02:00
|
|
|
|
2011-06-04 01:53:30 +02:00
|
|
|
return self;
|
2010-10-15 19:18:34 +02:00
|
|
|
}
|
|
|
|
|
2011-08-10 20:32:53 +02:00
|
|
|
- (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 tdb_wrap *) newIndexing
|
2010-10-01 20:54:30 +02:00
|
|
|
{
|
2012-10-03 17:25:47 +02:00
|
|
|
NSString *idNbr, *uri;
|
2010-12-13 17:54:32 +01:00
|
|
|
NSArray *keys;
|
|
|
|
NSUInteger count, max;
|
|
|
|
|
2011-06-04 01:53:30 +02:00
|
|
|
if ((self = [self init]))
|
2010-10-01 22:35:29 +02:00
|
|
|
{
|
2011-08-10 20:32:53 +02:00
|
|
|
ASSIGN (username, newUsername);
|
2011-06-04 01:53:30 +02:00
|
|
|
indexing = newIndexing;
|
2011-10-17 17:29:14 +02:00
|
|
|
(void) talloc_reference (memCtx, newIndexing);
|
2011-06-04 01:53:30 +02:00
|
|
|
tdb_traverse_read (indexing->tdb, MAPIStoreMappingTDBTraverse, mapping);
|
2010-12-13 17:54:32 +01:00
|
|
|
keys = [mapping allKeys];
|
|
|
|
max = [keys count];
|
|
|
|
for (count = 0; count < max; count++)
|
|
|
|
{
|
|
|
|
idNbr = [keys objectAtIndex: count];
|
|
|
|
uri = [mapping objectForKey: idNbr];
|
2011-09-04 18:25:32 +02:00
|
|
|
//[self logWithFormat: @"preregistered id '%@' for url '%@'", idNbr, uri];
|
2010-12-13 17:54:32 +01:00
|
|
|
[reverseMapping setObject: idNbr forKey: uri];
|
|
|
|
}
|
2011-09-04 18:25:32 +02:00
|
|
|
|
|
|
|
//[self logWithFormat: @"Complete mapping: %@ \nComplete reverse mapping: %@", mapping, reverseMapping];
|
2010-10-01 22:35:29 +02:00
|
|
|
}
|
2010-10-13 23:40:50 +02:00
|
|
|
|
2010-10-01 22:35:29 +02:00
|
|
|
return self;
|
2010-10-01 20:54:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void) dealloc
|
|
|
|
{
|
2011-08-10 20:32:53 +02:00
|
|
|
[username release];
|
2010-10-01 22:35:29 +02:00
|
|
|
[mapping release];
|
|
|
|
[reverseMapping release];
|
2011-10-17 17:29:14 +02:00
|
|
|
talloc_free (memCtx);
|
2010-10-01 22:35:29 +02:00
|
|
|
[super dealloc];
|
2010-10-01 20:54:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
- (NSString *) urlFromID: (uint64_t) idNbr
|
|
|
|
{
|
2012-10-03 17:25:47 +02:00
|
|
|
return [mapping objectForKey: MAPIStoreMappingKeyFromId (idNbr)];
|
2010-10-01 20:54:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
- (uint64_t) idFromURL: (NSString *) url
|
|
|
|
{
|
2012-10-03 17:25:47 +02:00
|
|
|
id key;
|
2010-10-01 22:35:29 +02:00
|
|
|
uint64_t idNbr;
|
2011-06-04 01:53:30 +02:00
|
|
|
|
2012-10-03 17:25:47 +02:00
|
|
|
key = [reverseMapping objectForKey: url];
|
|
|
|
if (key)
|
|
|
|
idNbr = [key unsignedLongLongValue];
|
2010-10-01 22:35:29 +02:00
|
|
|
else
|
|
|
|
idNbr = NSNotFound;
|
2010-10-13 17:30:01 +02:00
|
|
|
|
2010-10-01 22:35:29 +02:00
|
|
|
return idNbr;
|
2010-10-01 20:54:30 +02:00
|
|
|
}
|
|
|
|
|
2012-08-13 05:55:48 +02:00
|
|
|
- (void) _updateFolderWithURL: (NSString *) oldURL
|
|
|
|
withURL: (NSString *) urlString
|
|
|
|
{
|
|
|
|
NSArray *allKeys;
|
|
|
|
NSUInteger count, max;
|
2012-08-13 19:48:57 +02:00
|
|
|
NSString *currentKey, *newKey;
|
2012-10-03 17:25:47 +02:00
|
|
|
id idKey;
|
2012-08-13 05:55:48 +02:00
|
|
|
TDB_DATA key, dbuf;
|
|
|
|
|
2012-08-16 18:30:58 +02:00
|
|
|
[oldURL retain];
|
|
|
|
|
2012-08-13 05:55:48 +02:00
|
|
|
allKeys = [reverseMapping allKeys];
|
|
|
|
max = [allKeys count];
|
|
|
|
for (count = 0; count < max; count++)
|
|
|
|
{
|
|
|
|
currentKey = [allKeys objectAtIndex: count];
|
|
|
|
if ([currentKey hasPrefix: oldURL])
|
|
|
|
{
|
2012-08-13 19:48:57 +02:00
|
|
|
newKey = [currentKey stringByReplacingPrefix: oldURL
|
|
|
|
withPrefix: urlString];
|
2012-08-13 05:55:48 +02:00
|
|
|
|
|
|
|
idKey = [reverseMapping objectForKey: currentKey];
|
|
|
|
[mapping setObject: newKey forKey: idKey];
|
|
|
|
[reverseMapping setObject: idKey forKey: newKey];
|
|
|
|
[reverseMapping removeObjectForKey: currentKey];
|
|
|
|
|
|
|
|
/* update the record in the indexing database */
|
|
|
|
key.dptr = (unsigned char *) talloc_asprintf (NULL, "0x%.16"PRIx64,
|
|
|
|
(uint64_t) [idKey unsignedLongLongValue]);
|
|
|
|
key.dsize = strlen ((const char *) key.dptr);
|
|
|
|
|
|
|
|
dbuf.dptr = (unsigned char *) talloc_strdup (NULL,
|
|
|
|
[newKey UTF8String]);
|
|
|
|
dbuf.dsize = strlen ((const char *) dbuf.dptr);
|
|
|
|
tdb_store (indexing->tdb, key, dbuf, TDB_MODIFY);
|
|
|
|
talloc_free (key.dptr);
|
|
|
|
talloc_free (dbuf.dptr);
|
|
|
|
}
|
|
|
|
}
|
2012-08-16 18:30:58 +02:00
|
|
|
|
|
|
|
[oldURL release];
|
2012-08-13 05:55:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void) updateID: (uint64_t) idNbr
|
|
|
|
withURL: (NSString *) urlString
|
|
|
|
{
|
|
|
|
NSString *oldURL;
|
2012-10-03 17:25:47 +02:00
|
|
|
id idKey;
|
2012-08-13 05:55:48 +02:00
|
|
|
TDB_DATA key, dbuf;
|
|
|
|
|
2012-10-03 17:25:47 +02:00
|
|
|
idKey = MAPIStoreMappingKeyFromId (idNbr);
|
2012-08-13 05:55:48 +02:00
|
|
|
oldURL = [mapping objectForKey: idKey];
|
|
|
|
if (oldURL)
|
|
|
|
{
|
|
|
|
if ([oldURL hasSuffix: @"/"]) /* is container ? */
|
|
|
|
{
|
|
|
|
if (![urlString hasSuffix: @"/"])
|
|
|
|
[NSException raise: NSInvalidArgumentException
|
|
|
|
format: @"a container url must have an ending '/'"];
|
|
|
|
tdb_transaction_start (indexing->tdb);
|
|
|
|
[self _updateFolderWithURL: oldURL withURL: urlString];
|
|
|
|
tdb_transaction_commit (indexing->tdb);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ([urlString hasSuffix: @"/"])
|
|
|
|
[NSException raise: NSInvalidArgumentException
|
|
|
|
format: @"a leaf url must not have an ending '/'"];
|
|
|
|
[mapping setObject: urlString forKey: idKey];
|
|
|
|
[reverseMapping setObject: idKey forKey: urlString];
|
|
|
|
[reverseMapping removeObjectForKey: oldURL];
|
|
|
|
|
|
|
|
/* update the record in the indexing database */
|
|
|
|
key.dptr = (unsigned char *) talloc_asprintf(NULL, "0x%.16"PRIx64, idNbr);
|
|
|
|
key.dsize = strlen((const char *) key.dptr);
|
|
|
|
|
|
|
|
dbuf.dptr = (unsigned char *) talloc_strdup (NULL, [urlString UTF8String]);
|
|
|
|
dbuf.dsize = strlen((const char *) dbuf.dptr);
|
|
|
|
tdb_store (indexing->tdb, key, dbuf, TDB_MODIFY);
|
|
|
|
talloc_free (key.dptr);
|
|
|
|
talloc_free (dbuf.dptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-01 20:54:30 +02:00
|
|
|
- (BOOL) registerURL: (NSString *) urlString
|
|
|
|
withID: (uint64_t) idNbr
|
|
|
|
{
|
2012-10-03 17:25:47 +02:00
|
|
|
id idKey;
|
2010-10-01 22:35:29 +02:00
|
|
|
BOOL rc;
|
2011-06-04 01:53:30 +02:00
|
|
|
TDB_DATA key, dbuf;
|
2010-10-01 22:35:29 +02:00
|
|
|
|
2012-10-03 17:25:47 +02:00
|
|
|
idKey = MAPIStoreMappingKeyFromId (idNbr);
|
2010-10-01 22:35:29 +02:00
|
|
|
if ([mapping objectForKey: idKey]
|
|
|
|
|| [reverseMapping objectForKey: urlString])
|
|
|
|
{
|
2011-06-04 01:53:30 +02:00
|
|
|
[self errorWithFormat:
|
|
|
|
@"attempt to double register an entry ('%@', %lld,"
|
|
|
|
@" 0x%.16"PRIx64")",
|
|
|
|
urlString, idNbr, idNbr];
|
2010-10-01 22:35:29 +02:00
|
|
|
rc = NO;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
[mapping setObject: urlString forKey: idKey];
|
|
|
|
[reverseMapping setObject: idKey forKey: urlString];
|
|
|
|
rc = YES;
|
2011-10-04 00:21:36 +02:00
|
|
|
// [self logWithFormat: @"registered url '%@' with id %lld (0x%.16"PRIx64")",
|
|
|
|
// urlString, idNbr, idNbr];
|
2011-06-04 01:53:30 +02:00
|
|
|
|
|
|
|
/* Add the record given its fid and mapistore_uri */
|
|
|
|
key.dptr = (unsigned char *) talloc_asprintf(NULL, "0x%.16"PRIx64, idNbr);
|
|
|
|
key.dsize = strlen((const char *) key.dptr);
|
|
|
|
|
|
|
|
dbuf.dptr = (unsigned char *) talloc_strdup(NULL, [urlString UTF8String]);
|
|
|
|
dbuf.dsize = strlen((const char *) dbuf.dptr);
|
|
|
|
tdb_store (indexing->tdb, key, dbuf, TDB_INSERT);
|
|
|
|
talloc_free (key.dptr);
|
|
|
|
talloc_free (dbuf.dptr);
|
2010-10-01 22:35:29 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
2010-10-01 20:54:30 +02:00
|
|
|
}
|
|
|
|
|
2012-10-12 23:44:26 +02:00
|
|
|
- (void) registerURLs: (NSArray *) urlStrings
|
2012-10-13 05:27:05 +02:00
|
|
|
withIDs: (NSArray *) idNbrs
|
2012-10-12 23:44:26 +02:00
|
|
|
{
|
2012-10-13 05:27:05 +02:00
|
|
|
uint64_t count, max, newID;
|
2012-10-12 23:44:26 +02:00
|
|
|
|
|
|
|
max = [urlStrings count];
|
2012-10-13 05:27:05 +02:00
|
|
|
if (max == [idNbrs count])
|
2012-10-12 23:44:26 +02:00
|
|
|
{
|
|
|
|
tdb_transaction_start (indexing->tdb);
|
|
|
|
for (count = 0; count < max; count++)
|
2012-10-13 05:27:05 +02:00
|
|
|
{
|
|
|
|
newID = [[idNbrs objectAtIndex: count]
|
|
|
|
unsignedLongLongValue];
|
|
|
|
[self registerURL: [urlStrings objectAtIndex: count]
|
|
|
|
withID: newID];
|
|
|
|
}
|
2012-10-12 23:44:26 +02:00
|
|
|
tdb_transaction_commit (indexing->tdb);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
[NSException raise: NSInvalidArgumentException
|
|
|
|
format: @"number of urls and ids do not match"];
|
|
|
|
}
|
|
|
|
|
2011-02-03 23:09:41 +01:00
|
|
|
- (void) unregisterURLWithID: (uint64_t) idNbr
|
|
|
|
{
|
|
|
|
NSString *urlString;
|
2012-10-03 17:25:47 +02:00
|
|
|
id idKey;
|
2011-09-14 20:33:44 +02:00
|
|
|
TDB_DATA key;
|
2011-02-03 23:09:41 +01:00
|
|
|
|
2012-10-03 17:25:47 +02:00
|
|
|
idKey = MAPIStoreMappingKeyFromId (idNbr);
|
2011-02-03 23:09:41 +01:00
|
|
|
urlString = [mapping objectForKey: idKey];
|
2011-10-03 22:55:02 +02:00
|
|
|
if (urlString)
|
|
|
|
{
|
2011-10-04 00:21:36 +02:00
|
|
|
// [self logWithFormat: @"unregistering url '%@' with id %lld (0x%.16"PRIx64")",
|
|
|
|
// urlString, idNbr, idNbr];
|
2011-10-03 22:55:02 +02:00
|
|
|
[reverseMapping removeObjectForKey: urlString];
|
|
|
|
[mapping removeObjectForKey: idKey];
|
2011-09-14 20:33:44 +02:00
|
|
|
|
2011-10-03 22:55:02 +02:00
|
|
|
/* We hard-delete the entry from the indexing database */
|
|
|
|
key.dptr = (unsigned char *) talloc_asprintf(NULL, "0x%.16"PRIx64, idNbr);
|
|
|
|
key.dsize = strlen((const char *) key.dptr);
|
2011-09-14 20:33:44 +02:00
|
|
|
|
2011-10-03 22:55:02 +02:00
|
|
|
tdb_delete(indexing->tdb, key);
|
|
|
|
talloc_free(key.dptr);
|
|
|
|
}
|
2011-02-03 23:09:41 +01:00
|
|
|
}
|
|
|
|
|
2010-10-01 20:54:30 +02:00
|
|
|
@end
|