CKKSOutgoingQueueEntry.m [plain text]
/*
* Copyright (c) 2016 Apple Inc. All Rights Reserved.
*
* @APPLE_LICENSE_HEADER_START@
*
* This file contains Original Code and/or Modifications of Original Code
* as defined in and that are subject to the Apple Public Source License
* Version 2.0 (the 'License'). You may not use this file except in
* compliance with the License. Please obtain a copy of the License at
* http://www.opensource.apple.com/apsl/ and read it before using this
* file.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
* INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
* Please see the License for the specific language governing rights and
* limitations under the License.
*
* @APPLE_LICENSE_HEADER_END@
*/
#include <AssertMacros.h>
#import <Foundation/Foundation.h>
#import "CKKSKeychainView.h"
#include <Security/SecItemPriv.h>
#include <utilities/SecDb.h>
#include <securityd/SecDbItem.h>
#include <securityd/SecItemSchema.h>
#if OCTAGON
#import <CloudKit/CloudKit.h>
#import "CKKSOutgoingQueueEntry.h"
#import "CKKSItemEncrypter.h"
#import "CKKSKey.h"
@implementation CKKSOutgoingQueueEntry
- (NSString*)description {
return [NSString stringWithFormat: @"< NSStringFromClass([self class]),
self.item.zoneID.zoneName,
self.action,
self.item.uuid,
self.state];
}
- (instancetype) initWithCKKSItem:(CKKSItem*) item
action: (NSString*) action
state: (NSString*) state
waitUntil: (NSDate*) waitUntil
accessGroup: (NSString*) accessgroup
{
if((self = [super init])) {
_item = item;
_action = action;
_state = state;
_accessgroup = accessgroup;
_waitUntil = waitUntil;
}
return self;
}
- (BOOL)isEqual: (id) object {
if(![object isKindOfClass:[CKKSOutgoingQueueEntry class]]) {
return NO;
}
CKKSOutgoingQueueEntry* obj = (CKKSOutgoingQueueEntry*) object;
return ([self.item isEqual: obj.item] &&
[self.action isEqual: obj.action] &&
[self.state isEqual: obj.state] &&
((self.waitUntil == nil && obj.waitUntil == nil) || (fabs([self.waitUntil timeIntervalSinceDate: obj.waitUntil]) < 1)) &&
[self.accessgroup isEqual: obj.accessgroup] &&
true) ? YES : NO;
}
+ (instancetype)withItem: (SecDbItemRef) item action: (NSString*) action ckks:(CKKSKeychainView*) ckks error: (NSError * __autoreleasing *) error {
CFErrorRef cferror = NULL;
CKKSKey* key = nil;
NSString* uuid = nil;
NSString* accessgroup = nil;
NSInteger newGenerationCount = -1;
NSMutableDictionary* objd = nil;
key = [ckks keyForItem: item error:error];
if(!key) {
return nil;
}
objd = (__bridge_transfer NSMutableDictionary*) SecDbItemCopyPListWithMask(item, kSecDbSyncFlag, &cferror);
if(!objd) {
SecTranslateError(error, cferror);
return nil;
}
// Object classes aren't in the item plist, set them specifically
[objd setObject: (__bridge NSString*) item->class->name forKey: (__bridge NSString*) kSecClass];
uuid = (__bridge_transfer NSString*) CFRetain(SecDbItemGetValue(item, &v10itemuuid, &cferror));
if(!uuid || cferror) {
SecTranslateError(error, cferror);
return nil;
}
if([uuid isKindOfClass:[NSNull class]]) {
NSError* localerror = [NSError errorWithDomain:@"securityd"
code:CKKSNoUUIDOnItem
userInfo:@{NSLocalizedDescriptionKey: @"UUID not found in object"}];
secerror("ckksitem: couldn't fetch UUID: if(error) {
*error = localerror;
}
return nil;
}
accessgroup = (__bridge_transfer NSString*) CFRetain(SecDbItemGetValue(item, &v6agrp, &cferror));
if(!accessgroup || cferror) {
SecTranslateError(error, cferror);
return nil;
}
if([accessgroup isKindOfClass:[NSNull class]]) {
// That's okay; this is only used for rate limiting.
secerror("ckksitem: couldn't fetch accessgroup: accessgroup = @"no-group";
}
CKKSMirrorEntry* ckme = [CKKSMirrorEntry tryFromDatabase:uuid zoneID:ckks.zoneID error:error];
// The action this change should be depends on any existing pending action, if any
// Particularly, we need to coalesce (existing action, new action) to:
// (add, modify) => add
// (add, delete) => no-op
// (delete, add) => modify
NSString* actualAction = action;
CKKSOutgoingQueueEntry* existingOQE = [CKKSOutgoingQueueEntry tryFromDatabase:uuid state:SecCKKSStateNew zoneID:ckks.zoneID error:error];
if(existingOQE) {
if([existingOQE.action isEqual: SecCKKSActionAdd]) {
if([action isEqual:SecCKKSActionModify]) {
actualAction = SecCKKSActionAdd;
} else if([action isEqual:SecCKKSActionDelete]) {
// we're deleting an add. If there's a ckme, keep as a delete
if(!ckme) {
// Otherwise, remove from outgoingqueue and don't make a new OQE.
[existingOQE deleteFromDatabase:error];
return nil;
}
}
}
if([existingOQE.action isEqual: SecCKKSActionDelete] && [action isEqual:SecCKKSActionAdd]) {
actualAction = SecCKKSActionModify;
}
}
newGenerationCount = ckme ? ckme.item.generationCount : (NSInteger) 0; // TODO: this is wrong
// Pull out any unencrypted fields
NSNumber* pcsServiceIdentifier = objd[(id)kSecAttrPCSPlaintextServiceIdentifier];
objd[(id)kSecAttrPCSPlaintextServiceIdentifier] = nil;
NSData* pcsPublicKey = objd[(id)kSecAttrPCSPlaintextPublicKey];
objd[(id)kSecAttrPCSPlaintextPublicKey] = nil;
NSData* pcsPublicIdentity = objd[(id)kSecAttrPCSPlaintextPublicIdentity];
objd[(id)kSecAttrPCSPlaintextPublicIdentity] = nil;
CKKSItem* baseitem = [[CKKSItem alloc] initWithUUID:uuid
parentKeyUUID:key.uuid
zoneID:ckks.zoneID
encodedCKRecord:nil
encItem:nil
wrappedkey:nil
generationCount:newGenerationCount
encver:currentCKKSItemEncryptionVersion
plaintextPCSServiceIdentifier:pcsServiceIdentifier
plaintextPCSPublicKey:pcsPublicKey
plaintextPCSPublicIdentity:pcsPublicIdentity];
CKKSItem* encryptedItem = [CKKSItemEncrypter encryptCKKSItem:baseitem
dataDictionary:objd
updatingCKKSItem:ckme.item
parentkey:key
error:error];
if(!encryptedItem) {
return nil;
}
return [[CKKSOutgoingQueueEntry alloc] initWithCKKSItem:encryptedItem
action:actualAction
state:SecCKKSStateNew
waitUntil:nil
accessGroup:accessgroup];
}
#pragma mark - Property access to underlying CKKSItem
-(NSString*)uuid {
return self.item.uuid;
}
-(void)setUuid:(NSString *)uuid {
self.item.uuid = uuid;
}
#pragma mark - Database Operations
+ (instancetype) fromDatabase: (NSString*) uuid state: (NSString*) state zoneID:(CKRecordZoneID*)zoneID error: (NSError * __autoreleasing *) error {
return [self fromDatabaseWhere: @{@"UUID": CKKSNilToNSNull(uuid), @"state": CKKSNilToNSNull(state), @"ckzone":CKKSNilToNSNull(zoneID.zoneName)} error: error];
}
+ (instancetype) tryFromDatabase: (NSString*) uuid zoneID:(CKRecordZoneID*)zoneID error: (NSError * __autoreleasing *) error {
return [self tryFromDatabaseWhere: @{@"UUID": CKKSNilToNSNull(uuid), @"ckzone":CKKSNilToNSNull(zoneID.zoneName)} error: error];
}
+ (instancetype) tryFromDatabase: (NSString*) uuid state: (NSString*) state zoneID:(CKRecordZoneID*)zoneID error: (NSError * __autoreleasing *) error {
return [self tryFromDatabaseWhere: @{@"UUID": CKKSNilToNSNull(uuid), @"state":CKKSNilToNSNull(state), @"ckzone":CKKSNilToNSNull(zoneID.zoneName)} error: error];
}
+ (NSArray<CKKSOutgoingQueueEntry*>*) fetch:(ssize_t) n state: (NSString*) state zoneID:(CKRecordZoneID*)zoneID error: (NSError * __autoreleasing *) error {
return [self fetch:n where: @{@"state":CKKSNilToNSNull(state), @"ckzone":CKKSNilToNSNull(zoneID.zoneName)} error:error];
}
+ (NSArray<CKKSOutgoingQueueEntry*>*) allInState: (NSString*) state zoneID:(CKRecordZoneID*)zoneID error: (NSError * __autoreleasing *) error {
return [self allWhere: @{@"state":CKKSNilToNSNull(state), @"ckzone":CKKSNilToNSNull(zoneID.zoneName)} error:error];
}
#pragma mark - CKKSSQLDatabaseObject methods
+ (NSString*)sqlTable {
return @"outgoingqueue";
}
+ (NSArray<NSString*>*)sqlColumns {
return [[CKKSItem sqlColumns] arrayByAddingObjectsFromArray: @[@"action", @"state", @"waituntil", @"accessgroup"]];
}
- (NSDictionary<NSString*,NSString*>*)whereClauseToFindSelf {
return @{@"UUID": self.uuid, @"state": self.state, @"ckzone":self.item.zoneID.zoneName};
}
- (NSDictionary<NSString*,NSString*>*)sqlValues {
NSISO8601DateFormatter* dateFormat = [[NSISO8601DateFormatter alloc] init];
NSMutableDictionary* values = [[self.item sqlValues] mutableCopy];
values[@"action"] = self.action;
values[@"state"] = self.state;
values[@"waituntil"] = CKKSNilToNSNull(self.waitUntil ? [dateFormat stringFromDate: self.waitUntil] : nil);
values[@"accessgroup"] = self.accessgroup;
return values;
}
+ (instancetype)fromDatabaseRow: (NSDictionary*) row {
NSISO8601DateFormatter* dateFormat = [[NSISO8601DateFormatter alloc] init];
return [[CKKSOutgoingQueueEntry alloc] initWithCKKSItem:[CKKSItem fromDatabaseRow: row]
action:row[@"action"]
state:row[@"state"]
waitUntil:[row[@"waituntil"] isEqual: [NSNull null]] ? nil : [dateFormat dateFromString: row[@"waituntil"]]
accessGroup:row[@"accessgroup"]];
}
+ (NSDictionary<NSString*,NSNumber*>*)countsByState:(CKRecordZoneID*)zoneID error: (NSError * __autoreleasing *) error {
NSMutableDictionary* results = [[NSMutableDictionary alloc] init];
[CKKSSQLDatabaseObject queryDatabaseTable: [[self class] sqlTable]
where: @{@"ckzone": CKKSNilToNSNull(zoneID.zoneName)}
columns: @[@"state", @"count(rowid)"]
groupBy: @[@"state"]
orderBy:nil
limit: -1
processRow: ^(NSDictionary* row) {
results[row[@"state"]] = [NSNumber numberWithInteger: [row[@"count(rowid)"] integerValue]];
}
error: error];
return results;
}
@end
#endif