2010-12-13 17:49:54 +01:00
|
|
|
/* NSData+MAPIStore.m - this file is part of SOGo
|
2010-10-14 18:36:53 +02:00
|
|
|
*
|
2012-08-17 21:04:57 +02:00
|
|
|
* Copyright (C) 2010-2012 Inverse inc.
|
2010-10-14 18:36:53 +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-14 18:36:53 +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-08-21 22:39:46 +02:00
|
|
|
#import <NGExtensions/NSObject+Logs.h>
|
|
|
|
|
2015-07-20 11:17:00 +02:00
|
|
|
#import "MAPIStoreTypes.h"
|
2012-10-06 20:19:34 +02:00
|
|
|
#import "NSObject+MAPIStore.h"
|
2012-03-16 21:51:51 +01:00
|
|
|
#import "NSString+MAPIStore.h"
|
|
|
|
|
2010-10-14 18:36:53 +02:00
|
|
|
#import "NSData+MAPIStore.h"
|
|
|
|
|
|
|
|
#undef DEBUG
|
|
|
|
#include <stdbool.h>
|
2015-07-20 11:17:00 +02:00
|
|
|
#include <libmapi/libmapi.h>
|
2010-10-14 18:36:53 +02:00
|
|
|
#include <talloc.h>
|
2011-11-09 16:02:29 +01:00
|
|
|
#include <util/time.h>
|
|
|
|
#include <gen_ndr/exchange.h>
|
2010-10-14 18:36:53 +02:00
|
|
|
|
|
|
|
@implementation NSData (MAPIStoreDataTypes)
|
|
|
|
|
2010-11-23 15:49:46 +01:00
|
|
|
+ (id) dataWithBinary: (const struct Binary_r *) binData
|
|
|
|
{
|
|
|
|
return [NSData dataWithBytes: binData->lpb length: binData->cb];
|
|
|
|
}
|
|
|
|
|
2010-12-13 17:49:54 +01:00
|
|
|
- (struct Binary_r *) asBinaryInMemCtx: (void *) memCtx
|
|
|
|
{
|
|
|
|
struct Binary_r *binary;
|
|
|
|
|
|
|
|
binary = talloc_zero (memCtx, struct Binary_r);
|
|
|
|
binary->cb = [self length];
|
2012-10-06 20:19:34 +02:00
|
|
|
binary->lpb = (uint8_t *) [self bytes];
|
|
|
|
[self tallocWrapper: binary];
|
2010-12-13 17:49:54 +01:00
|
|
|
|
|
|
|
return binary;
|
|
|
|
}
|
|
|
|
|
2010-12-03 19:27:38 +01:00
|
|
|
+ (id) dataWithShortBinary: (const struct SBinary_short *) binData
|
|
|
|
{
|
|
|
|
return [NSData dataWithBytes: binData->lpb length: binData->cb];
|
|
|
|
}
|
|
|
|
|
2010-12-13 17:49:54 +01:00
|
|
|
- (struct SBinary_short *) asShortBinaryInMemCtx: (void *) memCtx
|
2010-10-14 18:36:53 +02:00
|
|
|
{
|
2010-12-13 17:49:54 +01:00
|
|
|
struct SBinary_short *binary;
|
2010-10-14 18:36:53 +02:00
|
|
|
|
2010-12-13 17:49:54 +01:00
|
|
|
binary = talloc_zero (memCtx, struct SBinary_short);
|
2010-10-14 18:36:53 +02:00
|
|
|
binary->cb = [self length];
|
2012-10-06 20:19:34 +02:00
|
|
|
binary->lpb = (uint8_t *) [self bytes];
|
|
|
|
[self tallocWrapper: binary];
|
2010-10-14 18:36:53 +02:00
|
|
|
|
|
|
|
return binary;
|
|
|
|
}
|
|
|
|
|
2010-12-30 15:47:27 +01:00
|
|
|
+ (id) dataWithFlatUID: (const struct FlatUID_r *) flatUID
|
|
|
|
{
|
|
|
|
return [NSData dataWithBytes: flatUID->ab length: 16];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (struct FlatUID_r *) asFlatUIDInMemCtx: (void *) memCtx
|
|
|
|
{
|
|
|
|
struct FlatUID_r *flatUID;
|
|
|
|
|
|
|
|
flatUID = talloc_zero (memCtx, struct FlatUID_r);
|
|
|
|
[self getBytes: flatUID->ab];
|
|
|
|
|
|
|
|
return flatUID;
|
|
|
|
}
|
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
static void _fillFlatUIDWithGUID (struct FlatUID_r *flatUID, const struct GUID *guid)
|
|
|
|
{
|
|
|
|
flatUID->ab[0] = (guid->time_low & 0xFF);
|
|
|
|
flatUID->ab[1] = ((guid->time_low >> 8) & 0xFF);
|
|
|
|
flatUID->ab[2] = ((guid->time_low >> 16) & 0xFF);
|
|
|
|
flatUID->ab[3] = ((guid->time_low >> 24) & 0xFF);
|
|
|
|
flatUID->ab[4] = (guid->time_mid & 0xFF);
|
|
|
|
flatUID->ab[5] = ((guid->time_mid >> 8) & 0xFF);
|
|
|
|
flatUID->ab[6] = (guid->time_hi_and_version & 0xFF);
|
|
|
|
flatUID->ab[7] = ((guid->time_hi_and_version >> 8) & 0xFF);
|
|
|
|
memcpy (flatUID->ab + 8, guid->clock_seq, sizeof (uint8_t) * 2);
|
|
|
|
memcpy (flatUID->ab + 10, guid->node, sizeof (uint8_t) * 6);
|
|
|
|
}
|
|
|
|
|
2010-12-30 15:47:27 +01:00
|
|
|
+ (id) dataWithGUID: (const struct GUID *) guid
|
|
|
|
{
|
|
|
|
struct FlatUID_r flatUID;
|
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
_fillFlatUIDWithGUID (&flatUID, guid);
|
2010-12-30 15:47:27 +01:00
|
|
|
|
|
|
|
return [self dataWithFlatUID: &flatUID];
|
|
|
|
}
|
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
- (void) _extractGUID: (struct GUID *) guid
|
2010-12-30 15:47:27 +01:00
|
|
|
{
|
2011-09-20 21:28:21 +02:00
|
|
|
uint8_t *bytes;
|
2010-12-30 15:47:27 +01:00
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
bytes = (uint8_t *) [self bytes];
|
2010-12-30 15:47:27 +01:00
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
guid->time_low = (bytes[3] << 24 | bytes[2] << 16
|
|
|
|
| bytes[1] << 8 | bytes[0]);
|
2010-12-30 15:47:27 +01:00
|
|
|
guid->time_mid = (bytes[5] << 8 | bytes[4]);
|
|
|
|
guid->time_hi_and_version = (bytes[7] << 8 | bytes[6]);
|
|
|
|
memcpy (guid->clock_seq, bytes + 8, sizeof (uint8_t) * 2);
|
|
|
|
memcpy (guid->node, bytes + 10, sizeof (uint8_t) * 6);
|
2011-09-20 21:28:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
- (struct GUID *) asGUIDInMemCtx: (void *) memCtx
|
|
|
|
{
|
|
|
|
struct GUID *guid;
|
|
|
|
|
|
|
|
guid = talloc_zero (memCtx, struct GUID);
|
|
|
|
[self _extractGUID: guid];
|
2010-12-30 15:47:27 +01:00
|
|
|
|
|
|
|
return guid;
|
|
|
|
}
|
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
+ (id) dataWithXID: (const struct XID *) xid
|
|
|
|
{
|
|
|
|
NSMutableData *xidData;
|
|
|
|
struct FlatUID_r flatUID;
|
|
|
|
|
2015-07-18 08:47:55 +02:00
|
|
|
_fillFlatUIDWithGUID (&flatUID, &xid->NameSpaceGuid);
|
2011-09-20 21:28:21 +02:00
|
|
|
|
2015-07-18 08:47:55 +02:00
|
|
|
xidData = [NSMutableData dataWithCapacity: 16 + xid->LocalId.length];
|
2011-09-20 21:28:21 +02:00
|
|
|
[xidData appendBytes: flatUID.ab length: 16];
|
2015-07-18 08:47:55 +02:00
|
|
|
[xidData appendBytes: xid->LocalId.data length: xid->LocalId.length];
|
2011-09-20 21:28:21 +02:00
|
|
|
|
|
|
|
return xidData;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (struct XID *) asXIDInMemCtx: (void *) memCtx
|
|
|
|
{
|
|
|
|
struct XID *xid;
|
2013-09-18 20:03:07 +02:00
|
|
|
uint8_t *bytes;
|
2011-09-20 21:28:21 +02:00
|
|
|
NSUInteger max;
|
|
|
|
|
|
|
|
max = [self length];
|
|
|
|
if (max > 16)
|
|
|
|
{
|
|
|
|
xid = talloc_zero (memCtx, struct XID);
|
|
|
|
|
2015-07-18 08:47:55 +02:00
|
|
|
[self _extractGUID: &xid->NameSpaceGuid];
|
2011-09-20 21:28:21 +02:00
|
|
|
|
2015-07-18 08:47:55 +02:00
|
|
|
xid->LocalId.length = max - 16;
|
2013-09-18 20:03:07 +02:00
|
|
|
|
|
|
|
bytes = (uint8_t *) [self bytes];
|
2015-07-18 08:47:55 +02:00
|
|
|
xid->LocalId.data = talloc_memdup (xid, (bytes+16), xid->LocalId.length);
|
2011-09-20 21:28:21 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
xid = NULL;
|
|
|
|
abort ();
|
|
|
|
}
|
|
|
|
|
|
|
|
return xid;
|
|
|
|
}
|
|
|
|
|
2015-07-20 11:17:00 +02:00
|
|
|
- (struct SizedXid *) asSizedXidArrayInMemCtx: (void *) memCtx
|
|
|
|
with: (uint32_t *) length
|
|
|
|
{
|
|
|
|
struct Binary_r bin;
|
|
|
|
struct SizedXid *sizedXIDArray;
|
|
|
|
|
|
|
|
bin.cb = [self length];
|
|
|
|
bin.lpb = (uint8_t *)[self bytes];
|
|
|
|
|
|
|
|
sizedXIDArray = get_SizedXidArray(memCtx, &bin, length);
|
|
|
|
if (!sizedXIDArray)
|
|
|
|
{
|
|
|
|
NSLog (@"Impossible to parse SizedXID array");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return sizedXIDArray;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (NSComparisonResult) compare: (NSData *) otherGlobCnt
|
|
|
|
{
|
|
|
|
uint64_t globCnt = 0, oGlobCnt = 0;
|
|
|
|
|
|
|
|
if ([self length] > 0)
|
|
|
|
globCnt = *(uint64_t *) [self bytes];
|
|
|
|
|
|
|
|
if ([otherGlobCnt length] > 0)
|
|
|
|
oGlobCnt = *(uint64_t *) [otherGlobCnt bytes];
|
|
|
|
|
|
|
|
return MAPICNCompare (globCnt, oGlobCnt, NULL);
|
|
|
|
}
|
|
|
|
|
2012-03-16 21:51:51 +01:00
|
|
|
+ (id) dataWithChangeKeyGUID: (NSString *) guidString
|
|
|
|
andCnt: (NSData *) globCnt;
|
|
|
|
{
|
|
|
|
NSMutableData *changeKey;
|
|
|
|
struct GUID guid;
|
|
|
|
|
|
|
|
changeKey = [NSMutableData dataWithCapacity: 16 + [globCnt length]];
|
|
|
|
|
|
|
|
[guidString extractGUID: &guid];
|
|
|
|
[changeKey appendData: [NSData dataWithGUID: &guid]];
|
|
|
|
[changeKey appendData: globCnt];
|
|
|
|
|
|
|
|
return changeKey;
|
|
|
|
}
|
|
|
|
|
2012-08-21 22:39:46 +02:00
|
|
|
- (void) hexDumpWithLineSize: (NSUInteger) lineSize
|
|
|
|
{
|
|
|
|
const char *bytes;
|
|
|
|
NSUInteger lineCount, count, max, charCount, charMax;
|
|
|
|
NSMutableString *line;
|
|
|
|
|
|
|
|
bytes = [self bytes];
|
|
|
|
max = [self length];
|
|
|
|
|
|
|
|
lineCount = 0;
|
|
|
|
for (count = 0; count < max; count++)
|
|
|
|
{
|
|
|
|
line = [NSMutableString stringWithFormat: @"%d: ", lineCount];
|
|
|
|
if (lineSize)
|
|
|
|
{
|
|
|
|
if ((max - count) > lineSize)
|
|
|
|
charMax = lineSize;
|
|
|
|
else
|
|
|
|
charMax = max - count;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
charMax = max;
|
|
|
|
for (charCount = 0; charCount < charMax; charCount++)
|
|
|
|
[line appendFormat: @" %.2x", *(bytes + count + charCount)];
|
|
|
|
[self logWithFormat: @" %@", line];
|
|
|
|
count += charMax;
|
|
|
|
lineCount++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation NSMutableData (MAPIStoreDataTypes)
|
|
|
|
|
|
|
|
- (void) appendUInt8: (uint8_t) value
|
|
|
|
{
|
|
|
|
[self appendBytes: (char *) &value length: 1];
|
|
|
|
}
|
|
|
|
|
2011-09-30 21:58:39 +02:00
|
|
|
- (void) appendUInt16: (uint16_t) value
|
|
|
|
{
|
|
|
|
NSUInteger count;
|
|
|
|
char bytes[2];
|
|
|
|
|
|
|
|
for (count = 0; count < 2; count++)
|
|
|
|
{
|
|
|
|
bytes[count] = value & 0xff;
|
|
|
|
value >>= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
[self appendBytes: bytes length: 2];
|
|
|
|
}
|
|
|
|
|
2011-09-20 21:28:21 +02:00
|
|
|
- (void) appendUInt32: (uint32_t) value
|
|
|
|
{
|
|
|
|
NSUInteger count;
|
|
|
|
char bytes[4];
|
|
|
|
|
|
|
|
for (count = 0; count < 4; count++)
|
|
|
|
{
|
|
|
|
bytes[count] = value & 0xff;
|
|
|
|
value >>= 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
[self appendBytes: bytes length: 4];
|
|
|
|
}
|
|
|
|
|
2010-10-14 18:36:53 +02:00
|
|
|
@end
|