2012-06-29 19:59:38 +02:00
|
|
|
/* MAPIStoreDBMessage.m - this file is part of SOGo
|
2011-02-07 18:49:47 +01:00
|
|
|
*
|
2012-08-17 21:04:57 +02:00
|
|
|
* Copyright (C) 2011-2012 Inverse inc
|
2011-02-07 18:49:47 +01: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
|
2011-08-12 17:02:01 +02:00
|
|
|
* the Free Software Foundation; either version 3, or (at your option)
|
2011-02-07 18:49:47 +01: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-06-01 17:15:30 +02:00
|
|
|
#import <Foundation/NSArray.h>
|
2012-06-29 19:59:38 +02:00
|
|
|
#import <Foundation/NSCalendarDate.h>
|
2015-07-20 11:17:00 +02:00
|
|
|
#import <Foundation/NSData.h>
|
2011-02-07 18:49:47 +01:00
|
|
|
#import <Foundation/NSDictionary.h>
|
2013-09-19 16:49:42 +02:00
|
|
|
#import <Foundation/NSString.h>
|
2011-08-01 20:07:00 +02:00
|
|
|
#import <Foundation/NSValue.h>
|
2011-09-27 21:26:12 +02:00
|
|
|
#import <NGExtensions/NSObject+Logs.h>
|
2011-02-07 18:49:47 +01:00
|
|
|
|
2011-07-28 00:04:49 +02:00
|
|
|
#import "MAPIStoreContext.h"
|
2011-06-07 02:17:46 +02:00
|
|
|
#import "MAPIStorePropertySelectors.h"
|
2012-06-29 19:59:38 +02:00
|
|
|
#import "SOGoMAPIDBMessage.h"
|
2011-02-07 18:49:47 +01:00
|
|
|
|
2012-06-29 19:59:38 +02:00
|
|
|
#import "MAPIStoreDBFolder.h"
|
|
|
|
#import "MAPIStoreDBMessage.h"
|
2011-10-26 23:33:51 +02:00
|
|
|
#import "MAPIStoreTypes.h"
|
2015-07-20 11:17:00 +02:00
|
|
|
#import "NSData+MAPIStore.h"
|
2012-06-29 19:59:38 +02:00
|
|
|
#import "NSObject+MAPIStore.h"
|
2013-09-19 16:49:42 +02:00
|
|
|
#import "NSString+MAPIStore.h"
|
2011-02-07 18:49:47 +01:00
|
|
|
|
2011-06-07 02:17:46 +02:00
|
|
|
#undef DEBUG
|
2011-07-28 00:04:49 +02:00
|
|
|
#include <mapistore/mapistore.h>
|
2011-06-07 02:17:46 +02:00
|
|
|
#include <mapistore/mapistore_errors.h>
|
|
|
|
|
2012-06-29 19:59:38 +02:00
|
|
|
@implementation MAPIStoreDBMessage
|
2011-02-07 18:49:47 +01:00
|
|
|
|
2015-09-18 21:45:00 +02:00
|
|
|
+ (enum mapistore_error) getAvailableProperties: (struct SPropTagArray **) propertiesP
|
|
|
|
inMemCtx: (TALLOC_CTX *) memCtx
|
2011-06-07 02:17:46 +02:00
|
|
|
{
|
|
|
|
struct SPropTagArray *properties;
|
|
|
|
NSUInteger count;
|
2015-09-18 21:45:00 +02:00
|
|
|
|
|
|
|
enum MAPITAGS faiProperties[] = {
|
|
|
|
0x68330048, /* PR_VIEW_CLSID */
|
|
|
|
0x68350102, /* PR_VIEW_STATE */
|
|
|
|
0x683c0102,
|
|
|
|
0x683d0040,
|
|
|
|
0x683e0102,
|
|
|
|
0x683f0102, /* PR_VIEW_VIEWTYPE_KEY */
|
|
|
|
0x68410003,
|
|
|
|
0x68420102,
|
|
|
|
0x68450102,
|
|
|
|
0x68460003,
|
|
|
|
0x7006001F, /* PR_VD_NAME_W */
|
|
|
|
0x70030003, /* PR_VD_FLAGS */
|
|
|
|
0x70070003 /* PR_VD_VERSION */
|
|
|
|
};
|
|
|
|
|
2015-02-19 15:00:27 +01:00
|
|
|
size_t faiSize = sizeof(faiProperties) / sizeof(enum MAPITAGS);
|
2011-06-07 02:17:46 +02:00
|
|
|
|
2011-07-12 23:47:20 +02:00
|
|
|
properties = talloc_zero (memCtx, struct SPropTagArray);
|
2015-02-19 15:00:27 +01:00
|
|
|
properties->cValues = MAPIStoreSupportedPropertiesCount + faiSize;
|
2011-10-27 17:59:50 +02:00
|
|
|
properties->aulPropTag = talloc_array (properties, enum MAPITAGS,
|
2015-02-19 15:00:27 +01:00
|
|
|
MAPIStoreSupportedPropertiesCount + faiSize);
|
2011-06-07 02:17:46 +02:00
|
|
|
|
|
|
|
for (count = 0; count < MAPIStoreSupportedPropertiesCount; count++)
|
|
|
|
properties->aulPropTag[count] = MAPIStoreSupportedProperties[count];
|
|
|
|
|
|
|
|
/* FIXME (hack): append a few undocumented properties that can be added to
|
|
|
|
FAI messages */
|
2015-02-19 15:00:27 +01:00
|
|
|
for (count = 0; count < faiSize; count++)
|
2012-06-29 19:59:38 +02:00
|
|
|
properties->aulPropTag[MAPIStoreSupportedPropertiesCount+count]
|
|
|
|
= faiProperties[count];
|
2011-06-07 02:17:46 +02:00
|
|
|
|
|
|
|
*propertiesP = properties;
|
|
|
|
|
|
|
|
return MAPISTORE_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-09-18 21:40:01 +02:00
|
|
|
- (enum mapistore_error) getAvailableProperties: (struct SPropTagArray **) propertiesP
|
|
|
|
inMemCtx: (TALLOC_CTX *) memCtx
|
|
|
|
{
|
|
|
|
return [[self class] getAvailableProperties: propertiesP
|
|
|
|
inMemCtx: memCtx];
|
|
|
|
}
|
|
|
|
|
2012-06-29 19:59:38 +02:00
|
|
|
- (id) initWithSOGoObject: (id) newSOGoObject
|
|
|
|
inContainer: (MAPIStoreObject *) newContainer
|
|
|
|
{
|
|
|
|
if ((self = [super initWithSOGoObject: newSOGoObject
|
|
|
|
inContainer: newContainer]))
|
|
|
|
{
|
|
|
|
[properties release];
|
|
|
|
properties = [newSOGoObject properties];
|
|
|
|
[properties retain];
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (uint64_t) objectVersion
|
|
|
|
{
|
|
|
|
NSNumber *versionNbr;
|
|
|
|
uint64_t objectVersion;
|
|
|
|
|
|
|
|
[(SOGoMAPIDBMessage *) sogoObject reloadIfNeeded];
|
|
|
|
versionNbr = [properties objectForKey: @"version"];
|
|
|
|
if (versionNbr)
|
2012-10-10 14:43:21 +02:00
|
|
|
objectVersion = (([versionNbr unsignedLongLongValue] >> 16)
|
|
|
|
& 0x0000ffffffffffffLL);
|
2012-06-29 19:59:38 +02:00
|
|
|
else
|
|
|
|
objectVersion = ULLONG_MAX;
|
|
|
|
|
|
|
|
return objectVersion;
|
|
|
|
}
|
|
|
|
|
2015-07-20 11:17:00 +02:00
|
|
|
- (void) _updatePredecessorChangeList
|
|
|
|
{
|
|
|
|
BOOL updated;
|
|
|
|
enum mapistore_error rc;
|
|
|
|
NSData *currentChangeList, *changeKey;
|
|
|
|
NSMutableArray *changeKeys;
|
|
|
|
NSMutableData *newChangeList;
|
|
|
|
NSUInteger count, len;
|
|
|
|
struct SizedXid *changes;
|
|
|
|
struct SPropValue property;
|
|
|
|
struct SRow aRow;
|
|
|
|
struct XID *currentChangeKey;
|
|
|
|
TALLOC_CTX *localMemCtx;
|
|
|
|
uint32_t nChanges;
|
|
|
|
|
|
|
|
localMemCtx = talloc_new (NULL);
|
|
|
|
if (!localMemCtx)
|
|
|
|
{
|
|
|
|
[self errorWithFormat: @"No more memory"];
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
changeKey = [self getReplicaKeyFromGlobCnt: [self objectVersion]];
|
|
|
|
|
|
|
|
currentChangeList = [properties objectForKey: MAPIPropertyKey (PidTagPredecessorChangeList)];
|
|
|
|
if (!currentChangeList)
|
|
|
|
{
|
|
|
|
/* Create a new PredecessorChangeList */
|
|
|
|
len = [changeKey length];
|
|
|
|
newChangeList = [NSMutableData dataWithCapacity: len + 1];
|
|
|
|
[newChangeList appendUInt8: len];
|
|
|
|
[newChangeList appendData: changeKey];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Update current predecessor change list with new change key */
|
|
|
|
changes = [currentChangeList asSizedXidArrayInMemCtx: localMemCtx
|
|
|
|
with: &nChanges];
|
|
|
|
|
|
|
|
updated = NO;
|
|
|
|
currentChangeKey = [changeKey asXIDInMemCtx: localMemCtx];
|
|
|
|
for (count = 0; count < nChanges && !updated; count++)
|
|
|
|
{
|
|
|
|
if (GUID_equal(&changes[count].XID.NameSpaceGuid, ¤tChangeKey->NameSpaceGuid))
|
|
|
|
{
|
|
|
|
NSData *globCnt, *oldGlobCnt;
|
|
|
|
oldGlobCnt = [NSData dataWithBytes: changes[count].XID.LocalId.data length: changes[count].XID.LocalId.length];
|
|
|
|
globCnt = [NSData dataWithBytes: currentChangeKey->LocalId.data length: currentChangeKey->LocalId.length];
|
|
|
|
if ([globCnt compare: oldGlobCnt] == NSOrderedDescending)
|
|
|
|
{
|
|
|
|
if ([globCnt length] != [oldGlobCnt length])
|
|
|
|
{
|
|
|
|
[self errorWithFormat: @"Cannot compare globcnt with different length: %@ and %@", globCnt, oldGlobCnt];
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
memcpy (changes[count].XID.LocalId.data, currentChangeKey->LocalId.data, currentChangeKey->LocalId.length);
|
|
|
|
updated = YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Serialise it */
|
|
|
|
changeKeys = [NSMutableArray array];
|
|
|
|
|
|
|
|
if (!updated)
|
|
|
|
[changeKeys addObject: changeKey];
|
|
|
|
|
|
|
|
for (count = 0; count < nChanges; count++)
|
|
|
|
{
|
|
|
|
changeKey = [NSData dataWithXID: &changes[count].XID];
|
|
|
|
[changeKeys addObject: changeKey];
|
|
|
|
}
|
|
|
|
|
|
|
|
[changeKeys sortUsingFunction: MAPIChangeKeyGUIDCompare context: localMemCtx];
|
|
|
|
|
|
|
|
newChangeList = [NSMutableData data];
|
|
|
|
len = [changeKeys count];
|
|
|
|
for (count = 0; count < len; count++)
|
|
|
|
{
|
|
|
|
changeKey = [changeKeys objectAtIndex: count];
|
|
|
|
[newChangeList appendUInt8: [changeKey length]];
|
|
|
|
[newChangeList appendData: changeKey];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ([newChangeList length] > 0)
|
|
|
|
{
|
|
|
|
property.ulPropTag = PidTagPredecessorChangeList;
|
|
|
|
property.value.bin = *[newChangeList asBinaryInMemCtx: localMemCtx];
|
|
|
|
aRow.cValues = 1;
|
|
|
|
aRow.lpProps = &property;
|
|
|
|
rc = [self addPropertiesFromRow: &aRow];
|
|
|
|
if (rc != MAPISTORE_SUCCESS)
|
|
|
|
[self errorWithFormat: @"Impossible to add a new predecessor change list: %d", rc];
|
|
|
|
}
|
|
|
|
|
|
|
|
talloc_free (localMemCtx);
|
|
|
|
}
|
|
|
|
|
2013-09-19 16:49:42 +02:00
|
|
|
//
|
|
|
|
// FIXME: how this can happen?
|
|
|
|
//
|
|
|
|
// We might get there if for some reasons, all classes weren't able
|
|
|
|
// to tell us the message class.
|
|
|
|
//
|
|
|
|
- (int) getPidTagMessageClass: (void **) data inMemCtx: (TALLOC_CTX *) memCtx
|
|
|
|
{
|
|
|
|
*data = [@"IPM.Note" asUnicodeInMemCtx: memCtx];
|
|
|
|
|
|
|
|
[self logWithFormat: @"METHOD '%s' - unable to determine message class. Falling back to IPM.Note", __FUNCTION__];
|
|
|
|
|
|
|
|
return MAPISTORE_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2012-06-29 19:59:38 +02:00
|
|
|
- (int) getProperties: (struct mapistore_property_data *) data
|
|
|
|
withTags: (enum MAPITAGS *) tags
|
|
|
|
andCount: (uint16_t) columnCount
|
|
|
|
inMemCtx: (TALLOC_CTX *) memCtx
|
|
|
|
{
|
|
|
|
[sogoObject reloadIfNeeded];
|
|
|
|
|
|
|
|
return [super getProperties: data
|
|
|
|
withTags: tags
|
|
|
|
andCount: columnCount
|
|
|
|
inMemCtx: memCtx];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (int) getProperty: (void **) data
|
|
|
|
withTag: (enum MAPITAGS) propTag
|
|
|
|
inMemCtx: (TALLOC_CTX *) memCtx
|
|
|
|
{
|
|
|
|
id value;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
value = [properties objectForKey: MAPIPropertyKey (propTag)];
|
|
|
|
if (value)
|
|
|
|
rc = [value getValue: data forTag: propTag inMemCtx: memCtx];
|
|
|
|
else
|
|
|
|
rc = [super getProperty: data withTag: propTag inMemCtx: memCtx];
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void) addProperties: (NSDictionary *) newNewProperties
|
|
|
|
{
|
|
|
|
[sogoObject reloadIfNeeded];
|
|
|
|
|
|
|
|
[super addProperties: newNewProperties];
|
|
|
|
}
|
|
|
|
|
2013-03-13 20:04:11 +01:00
|
|
|
- (void) save: (TALLOC_CTX *) memCtx
|
2011-02-07 18:49:47 +01:00
|
|
|
{
|
2011-07-28 00:04:49 +02:00
|
|
|
uint64_t newVersion;
|
|
|
|
|
2011-07-15 22:22:27 +02:00
|
|
|
if ([attachmentKeys count] > 0)
|
2011-10-26 17:15:17 +02:00
|
|
|
[properties setObject: attachmentParts forKey: @"attachments"];
|
2011-07-28 00:04:49 +02:00
|
|
|
|
2012-07-19 03:25:04 +02:00
|
|
|
newVersion = [[self context] getNewChangeNumber];
|
2011-10-26 17:15:17 +02:00
|
|
|
[properties setObject: [NSNumber numberWithUnsignedLongLong: newVersion]
|
2012-06-29 19:59:38 +02:00
|
|
|
forKey: @"version"];
|
2011-09-27 21:26:12 +02:00
|
|
|
|
2015-07-20 11:17:00 +02:00
|
|
|
/* Update PredecessorChangeList accordingly */
|
|
|
|
[self _updatePredecessorChangeList];
|
|
|
|
|
2011-10-26 17:15:17 +02:00
|
|
|
[self logWithFormat: @"%d props in dict", [properties count]];
|
2011-09-27 21:26:12 +02:00
|
|
|
|
2011-02-24 21:30:01 +01:00
|
|
|
[sogoObject save];
|
2011-06-07 02:17:46 +02:00
|
|
|
}
|
|
|
|
|
2011-12-01 23:53:11 +01:00
|
|
|
- (BOOL) _messageIsFreeBusy
|
2011-12-01 23:34:52 +01:00
|
|
|
{
|
|
|
|
NSString *msgClass;
|
|
|
|
|
2011-12-01 23:53:11 +01:00
|
|
|
/* This is a HACK until we figure out how to determine a message position in
|
|
|
|
the mailbox hierarchy.... (missing: folderid and role) */
|
2012-06-29 19:59:38 +02:00
|
|
|
msgClass = [properties
|
2011-12-01 23:34:52 +01:00
|
|
|
objectForKey: MAPIPropertyKey (PR_MESSAGE_CLASS_UNICODE)];
|
|
|
|
|
2011-12-01 23:53:11 +01:00
|
|
|
return [msgClass isEqualToString: @"IPM.Microsoft.ScheduleData.FreeBusy"];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: differentiate between the "Own" and "All" cases */
|
|
|
|
- (BOOL) subscriberCanReadMessage
|
|
|
|
{
|
2012-10-05 16:09:38 +02:00
|
|
|
return [(MAPIStoreFolder *) container subscriberCanReadMessages];
|
|
|
|
// || [self _messageIsFreeBusy]);
|
2011-12-01 23:53:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL) subscriberCanModifyMessage
|
|
|
|
{
|
2011-12-02 18:14:50 +01:00
|
|
|
return ((isNew
|
|
|
|
&& [(MAPIStoreFolder *) container subscriberCanCreateMessages])
|
|
|
|
|| (!isNew
|
2012-10-05 16:09:38 +02:00
|
|
|
&& [(MAPIStoreFolder *) container subscriberCanModifyMessages]));
|
|
|
|
// || [self _messageIsFreeBusy]);
|
2011-12-01 23:34:52 +01:00
|
|
|
}
|
|
|
|
|
2011-06-07 02:17:46 +02:00
|
|
|
- (NSDate *) creationTime
|
|
|
|
{
|
2012-06-29 19:59:38 +02:00
|
|
|
return [sogoObject creationDate];
|
2011-06-07 02:17:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
- (NSDate *) lastModificationTime
|
|
|
|
{
|
2012-06-29 19:59:38 +02:00
|
|
|
return [sogoObject lastModified];
|
2011-02-07 18:49:47 +01:00
|
|
|
}
|
|
|
|
|
2015-09-04 16:16:00 +02:00
|
|
|
- (enum mapistore_error) setReadFlag: (uint8_t) flag
|
|
|
|
{
|
|
|
|
/* Modify PidTagMessageFlags from SetMessageReadFlag and
|
|
|
|
SyncImportReadStateChanges ROPs */
|
|
|
|
NSNumber *flags;
|
|
|
|
uint32_t newFlag;
|
|
|
|
|
|
|
|
flags = [properties objectForKey: MAPIPropertyKey (PR_MESSAGE_FLAGS)];
|
|
|
|
if (flags)
|
|
|
|
{
|
|
|
|
newFlag = [flags unsignedLongValue];
|
|
|
|
if (flag & SUPPRESS_RECEIPT)
|
|
|
|
newFlag |= MSGFLAG_READ;
|
|
|
|
if (flag & CLEAR_RN_PENDING)
|
|
|
|
newFlag &= ~MSGFLAG_RN_PENDING;
|
|
|
|
if (flag & CLEAR_READ_FLAG)
|
|
|
|
newFlag &= ~MSGFLAG_READ;
|
|
|
|
if (flag & CLEAR_NRN_PENDING)
|
|
|
|
newFlag &= ~MSGFLAG_NRN_PENDING;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
newFlag = MSGFLAG_READ;
|
|
|
|
if (flag & CLEAR_READ_FLAG)
|
|
|
|
newFlag = 0x0;
|
|
|
|
}
|
|
|
|
[properties setObject: [NSNumber numberWithUnsignedLong: newFlag]
|
|
|
|
forKey: MAPIPropertyKey (PR_MESSAGE_FLAGS)];
|
|
|
|
|
|
|
|
return MAPISTORE_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2011-02-07 18:49:47 +01:00
|
|
|
@end
|