1 #include <CoreFoundation/CoreFoundation.h>
2 #include <CoreFoundation/CFRuntime.h>
4 #include <Security/SecureObjectSync/SOSAccount.h>
5 #include <Security/SecureObjectSync/SOSAccountPriv.h>
6 #include <Security/SecureObjectSync/SOSTransport.h>
7 #import <Security/SecureObjectSync/SOSTransportKeyParameter.h>
8 #import <Security/SecureObjectSync/SOSTransportCircleKVS.h>
9 #import <Security/SecureObjectSync/SOSTransportMessageIDS.h>
10 #import <Security/SecureObjectSync/SOSTransportMessageKVS.h>
11 #include <Security/SecureObjectSync/SOSKVSKeys.h>
12 #include <Security/SecureObjectSync/SOSPeerCoder.h>
13 #include <utilities/SecCFWrappers.h>
14 #include <Security/SecureObjectSync/SOSPeerInfoV2.h>
15 #import <Security/SecureObjectSync/SOSAccountTrustClassic+Circle.h>
16 #import <Security/SecureObjectSync/SOSAccountTrustClassic+Identity.h>
18 #include "SOSTransportTestTransports.h"
19 #include "SOSAccountTesting.h"
21 CFMutableArrayRef key_transports = NULL;
22 CFMutableArrayRef circle_transports = NULL;
23 CFMutableArrayRef message_transports = NULL;
26 //Mark Test Key Parameter Transport
29 @implementation CKKeyParameterTest
31 -(id) initWithAccount:(SOSAccount*) acct andName:(CFStringRef) n andCircleName:(CFStringRef) cN
35 self.name = CFRetainSafe(n);
36 self.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
38 self.circleName = CFRetainSafe(cN);
41 key_transports = CFArrayCreateMutable(kCFAllocatorDefault, 0, NULL);
42 CFArrayAppendValue(key_transports, (__bridge CFTypeRef)((CKKeyParameter*)self));
44 SOSRegisterTransportKeyParameter((CKKeyParameter*)self);
51 CFReleaseNull(self->_changes);
52 CFReleaseNull(self->_circleName);
56 - (void)setChanges:(CFMutableDictionaryRef)changes
58 CFRetainAssign(self->_changes, changes);
61 -(bool) SOSTransportKeyParameterHandleKeyParameterChanges:(CKKeyParameterTest*) transport data:(CFDataRef) data err:(CFErrorRef) error
63 SOSAccount* acct = transport.account;
64 return SOSAccountHandleParametersChange(acct, data, &error);
68 -(void) SOSTransportKeyParameterHandleNewAccount:(CKKeyParameterTest*) transport acct:(SOSAccount*) acct
72 CFArrayRemoveAllValue(key_transports, (__bridge CFTypeRef)(acct.key_transport));
74 if(message_transports){
75 CFArrayRemoveAllValue(message_transports, (__bridge CFTypeRef)acct.ids_message_transport);
76 CFArrayRemoveAllValue(message_transports, (__bridge CFTypeRef)acct.kvs_message_transport);
79 CFArrayRemoveAllValue(circle_transports, (__bridge CFTypeRef)(acct.circle_transport));
81 SOSAccountSetToNew(acct);
82 SOSAccountResetToTest(acct, transport.name);
85 CFStringRef SOSTransportKeyParameterTestGetName(CKKeyParameterTest* transport){
86 return transport.name;
89 void SOSTransportKeyParameterTestSetName(CKKeyParameterTest* transport, CFStringRef accountName){
90 transport.name = accountName;
93 SOSAccount* SOSTransportKeyParameterTestGetAccount(CKKeyParameterTest* transport){
94 return ((CKKeyParameter*)transport).account;
97 CFMutableDictionaryRef SOSTransportKeyParameterTestGetChanges(CKKeyParameterTest* transport){
98 return transport.changes;
101 void SOSTransportKeyParameterTestClearChanges(CKKeyParameterTest* transport){
102 transport.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
105 -(bool) SOSTransportKeyParameterPublishCloudParameters:(CKKeyParameterTest*) transport data:(CFDataRef)newParameters err:(CFErrorRef*) error
107 if(!transport.changes)
108 transport.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
110 CFDictionarySetValue(transport.changes, kSOSKVSKeyParametersKey, newParameters);
118 //MARK: Test Circle Transport
120 @implementation SOSCircleStorageTransportTest
121 @synthesize accountName = accountName;
128 CFStringRef SOSTransportCircleTestGetName(SOSCircleStorageTransportTest* transport){
129 return (__bridge CFStringRef)(transport.accountName);
131 void SOSTransportCircleTestSetName(SOSCircleStorageTransportTest* transport, CFStringRef name){
132 transport.accountName = nil;
133 transport.accountName = (__bridge NSString *)(name);
136 -(CFMutableDictionaryRef) SOSTransportCircleTestGetChanges
138 return (__bridge CFMutableDictionaryRef)(self.pending_changes);
141 void SOSTransportCircleTestClearChanges(SOSCircleStorageTransportTest* transport){
142 transport.pending_changes = [NSMutableDictionary dictionary];
145 -(id) initWithAccount:(SOSAccount *)acct andWithAccountName:(CFStringRef)acctName andCircleName:(CFStringRef)cName
150 self.accountName = (__bridge NSString *)(acctName);
151 self.circleName = (__bridge NSString*)cName;
152 self.pending_changes = [NSMutableDictionary dictionary];
153 if(!circle_transports)
154 circle_transports = CFArrayCreateMutable(kCFAllocatorDefault, 0, NULL);
155 CFArrayAppendValue(circle_transports, (__bridge CFTypeRef)self);
157 SOSRegisterTransportCircle((SOSCircleStorageTransportTest*)self);
162 -(bool) kvsRingFlushChanges:(CFErrorRef*) error
167 -(bool) kvsRingPostRing:(CFStringRef) ringName ring:(CFDataRef) ring err:(CFErrorRef *)error
169 CFStringRef ringKey = SOSRingKeyCreateWithName(ringName, error);
170 CFMutableDictionaryRef changes = [self SOSTransportCircleTestGetChanges];
171 CFDictionaryAddValue(changes, ringKey, ring);
172 CFReleaseNull(ringKey);
176 -(bool) kvssendDebugInfo:(CFStringRef) type debug:(CFTypeRef) debugInfo err:(CFErrorRef *)error
178 CFMutableDictionaryRef changes = [self SOSTransportCircleTestGetChanges];
179 CFDictionaryAddValue(changes, type, debugInfo);
183 -(bool) postRetirement:(CFStringRef)cName peer:(SOSPeerInfoRef)peer err:(CFErrorRef *)error
185 CFStringRef retirement_key = SOSRetirementKeyCreateWithCircleNameAndPeer(cName, SOSPeerInfoGetPeerID(peer));
186 CFDataRef retirement_data = SOSPeerInfoCopyEncodedData(peer, kCFAllocatorDefault, error);
189 [self testAddToChanges:retirement_key data:retirement_data];
191 CFReleaseNull(retirement_key);
192 CFReleaseNull(retirement_data);
196 -(bool) flushChanges:(CFErrorRef *)error
201 -(void) testAddToChanges:(CFStringRef) message_key data:(CFDataRef) message_data
203 if (self.pending_changes == NULL) {
204 self.pending_changes = [NSMutableDictionary dictionary];
206 if (message_data == NULL) {
207 [self.pending_changes setObject:[NSNull null] forKey:(__bridge NSString*)(message_key)];
209 [self.pending_changes setObject:(__bridge NSData*)message_data forKey:(__bridge NSString*)message_key];
211 secnotice("circle-changes", "Adding circle change %@ %@->%@", self.accountName, message_key, message_data);
214 -(void) SOSTransportCircleTestAddBulkToChanges:(CFDictionaryRef) updates
216 if (self.pending_changes == NULL) {
217 self.pending_changes = [[NSMutableDictionary alloc]initWithDictionary:(__bridge NSDictionary * _Nonnull)(updates)];
221 CFDictionaryForEach(updates, ^(const void *key, const void *value) {
222 [self.pending_changes setObject:(__bridge id _Nonnull)value forKey:(__bridge id _Nonnull)key];
228 -(bool) expireRetirementRecords:(CFDictionaryRef) retirements err:(CFErrorRef *)error
231 CFMutableDictionaryRef keysToWrite = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
233 CFDictionaryForEach(retirements, ^(const void *key, const void *value) {
234 if (isString(key) && isArray(value)) {
235 CFStringRef circle_name = (CFStringRef) key;
236 CFArrayRef retirees = (CFArrayRef) value;
238 CFArrayForEach(retirees, ^(const void *value) {
239 if (isString(value)) {
240 CFStringRef retiree_id = (CFStringRef) value;
242 CFStringRef kvsKey = SOSRetirementKeyCreateWithCircleNameAndPeer(circle_name, retiree_id);
244 CFDictionaryAddValue(keysToWrite, kvsKey, kCFNull);
246 CFReleaseSafe(kvsKey);
252 if(CFDictionaryGetCount(keysToWrite)) {
253 [self SOSTransportCircleTestAddBulkToChanges:keysToWrite];
255 CFReleaseNull(keysToWrite);
260 bool SOSTransportCircleTestRemovePendingChange(SOSCircleStorageTransportTest* transport, CFStringRef circleName, CFErrorRef *error){
261 CFStringRef circle_key = SOSCircleKeyCreateWithName(circleName, error);
263 [transport.pending_changes removeObjectForKey:(__bridge NSString*)circle_key];
264 CFReleaseNull(circle_key);
268 -(bool) postCircle:(CFStringRef)cName circleData:(CFDataRef)circle_data err:(CFErrorRef *)error
270 CFStringRef circle_key = SOSCircleKeyCreateWithName(cName, error);
272 [self testAddToChanges:circle_key data:circle_data];
273 CFReleaseNull(circle_key);
278 -(CFDictionaryRef)handleRetirementMessages:(CFMutableDictionaryRef) circle_retirement_messages_table err:(CFErrorRef *)error
280 return SOSAccountHandleRetirementMessages(self.account, circle_retirement_messages_table, error);
283 -(CFArrayRef)CF_RETURNS_RETAINED handleCircleMessagesAndReturnHandledCopy:(CFMutableDictionaryRef) circle_circle_messages_table err:(CFErrorRef *)error
285 CFMutableArrayRef handledKeys = CFArrayCreateMutableForCFTypes(kCFAllocatorDefault);
286 CFDictionaryForEach(circle_circle_messages_table, ^(const void *key, const void *value) {
287 CFErrorRef circleMessageError = NULL;
288 if (!SOSAccountHandleCircleMessage(self.account, key, value, &circleMessageError)) {
289 secerror("Error handling circle message %@ (%@): %@", key, value, circleMessageError);
292 CFStringRef circle_id = (CFStringRef) key;
293 CFArrayAppendValue(handledKeys, circle_id);
295 CFReleaseNull(circleMessageError);
301 SOSAccount* SOSTransportCircleTestGetAccount(SOSCircleStorageTransportTest* transport) {
302 return transport.account;
308 //MARK KVS Message Test Transport
313 @implementation SOSMessageKVSTest
316 -(id) initWithAccount:(SOSAccount*)acct andName:(CFStringRef)n andCircleName:(CFStringRef) cN
320 self.engine = SOSDataSourceFactoryGetEngineForDataSourceName(acct.factory, cN, NULL);
322 self.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
323 self.name = CFRetainSafe(n);
324 self.circleName = (__bridge NSString*)cN;
326 if(!message_transports)
327 message_transports = CFArrayCreateMutable(kCFAllocatorDefault, 0, NULL);
328 CFArrayAppendValue(message_transports, (__bridge const void *)((SOSMessageKVSTest*)self));
329 SOSRegisterTransportMessage((SOSMessageKVSTest*)self);
335 - (void)setChanges:(CFMutableDictionaryRef)changes
337 CFRetainAssign(self->_changes, changes);
340 -(CFIndex) SOSTransportMessageGetTransportType
344 -(CFStringRef) SOSTransportMessageGetCircleName
346 return (__bridge CFStringRef)circleName;
348 -(CFTypeRef) SOSTransportMessageGetEngine
352 -(SOSAccount*) SOSTransportMessageGetAccount
357 void SOSTransportMessageKVSTestSetName(SOSMessageKVSTest* transport, CFStringRef n)
362 CFStringRef SOSTransportMessageKVSTestGetName(SOSMessageKVSTest* transport)
364 return transport.name;
367 CFMutableDictionaryRef SOSTransportMessageKVSTestGetChanges(SOSMessageKVSTest* transport)
369 return transport.changes;
372 void SOSTransportMessageTestClearChanges(SOSMessageKVSTest* transport)
374 transport.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
377 static void SOSTransportMessageTestAddBulkToChanges(SOSMessageKVSTest* transport, CFDictionaryRef updates)
379 #ifndef __clang_analyzer__ // The analyzer thinks transport.changes is a leak, but I don't know why.
380 if (transport.changes == NULL) {
381 transport.changes = CFDictionaryCreateMutableCopy(kCFAllocatorDefault, CFDictionaryGetCount(updates), updates);
384 CFDictionaryForEach(updates, ^(const void *key, const void *value) {
385 CFDictionarySetValue(transport.changes, key, value);
388 #endif // __clang_analyzer__
391 static void SOSTransportMessageTestAddToChanges(SOSMessageKVSTest* transport, CFStringRef message_key, CFDataRef message_data)
393 if (transport.changes == NULL) {
394 transport.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
396 if (message_data == NULL) {
397 CFDictionarySetValue(transport.changes, message_key, kCFNull);
399 CFDictionarySetValue(transport.changes, message_key, message_data);
403 -(bool) SOSTransportMessageCleanupAfterPeerMessages:(SOSMessageKVSTest*) transport peers:(CFDictionaryRef)circle_to_peer_ids err:(CFErrorRef*) error
405 if(!transport.engine)
407 CFArrayRef enginePeers = SOSEngineGetPeerIDs((SOSEngineRef)transport.engine);
409 CFDictionaryForEach(circle_to_peer_ids, ^(const void *key, const void *value) {
410 if (isString(key) && isArray(value)) {
411 CFStringRef circle_name = (CFStringRef) key;
412 CFArrayRef peers_to_cleanup_after = (CFArrayRef) value;
414 CFArrayForEach(peers_to_cleanup_after, ^(const void *value) {
415 if (isString(value)) {
416 CFStringRef cleanup_id = (CFStringRef) value;
417 // TODO: Since the enginePeers list is not authorative (the Account is) this could inadvertently clean up active peers or leave behind stale peers
418 if (enginePeers) CFArrayForEach(enginePeers, ^(const void *value) {
419 if (isString(value)) {
420 CFStringRef in_circle_id = (CFStringRef) value;
422 CFStringRef kvsKey = SOSMessageKeyCreateWithCircleNameAndPeerNames(circle_name, cleanup_id, in_circle_id);
423 SOSTransportMessageTestAddToChanges(transport, kvsKey, NULL);
424 CFReleaseSafe(kvsKey);
426 kvsKey = SOSMessageKeyCreateWithCircleNameAndPeerNames(circle_name, in_circle_id, cleanup_id);
427 SOSTransportMessageTestAddToChanges(transport, kvsKey, NULL);
428 CFReleaseSafe(kvsKey);
437 return [transport SOSTransportMessageFlushChanges:transport err:error];
441 static bool sendToPeer(SOSMessageKVSTest* transport, CFStringRef circleName, CFStringRef peerID, CFDataRef message, CFErrorRef *error) {
443 NSString* myID = transport.account.peerID;
444 CFStringRef message_to_peer_key = SOSMessageKeyCreateFromTransportToPeer((SOSMessage*)transport, (__bridge CFStringRef) myID, peerID);
445 CFDictionaryRef a_message_to_a_peer = CFDictionaryCreateForCFTypes(NULL, message_to_peer_key, message, NULL);
447 SOSTransportMessageTestAddBulkToChanges(transport, a_message_to_a_peer);
448 CFReleaseNull(a_message_to_a_peer);
449 CFReleaseNull(message_to_peer_key);
454 -(bool) SOSTransportMessageSyncWithPeers:(SOSMessageKVSTest*) transport p:(CFSetRef) peers err:(CFErrorRef *)error
456 // Each entry is keyed by circle name and contains a list of peerIDs
458 __block bool result = true;
460 CFSetForEach(peers, ^(const void *value) {
461 CFStringRef peerID = asString(value, NULL);
464 SOSEngineWithPeerID((SOSEngineRef)transport.engine, peerID, error, ^(SOSPeerRef peer, SOSCoderRef coder, SOSDataSourceRef dataSource, SOSTransactionRef txn, bool *forceSaveState) {
465 SOSEnginePeerMessageSentCallback* sentCallback = NULL;
466 CFDataRef message_to_send = NULL;
467 bool ok = SOSPeerCoderSendMessageIfNeeded([transport SOSTransportMessageGetAccount], (SOSEngineRef)transport.engine, txn, peer, coder, &message_to_send, peerID, false, &sentCallback, error);
468 if (message_to_send) {
469 CFDictionaryRef peer_dict = CFDictionaryCreateForCFTypes(kCFAllocatorDefault, peerID, message_to_send, NULL);
470 CFDictionarySetValue(SOSTransportMessageKVSTestGetChanges(transport), (__bridge CFStringRef)self->circleName, peer_dict);
471 SOSEngineMessageCallCallback(sentCallback, ok);
472 CFReleaseSafe(peer_dict);
475 SOSEngineFreeMessageCallback(sentCallback);
476 CFReleaseSafe(message_to_send);
483 -(bool) SOSTransportMessageSendMessages:(SOSMessageKVSTest*) transport pm:(CFDictionaryRef) peer_messages err:(CFErrorRef *)error
485 __block bool result = true;
487 CFDictionaryForEach(peer_messages, ^(const void *key, const void *value) {
488 if (isString(key) && isData(value)) {
489 CFStringRef peerID = (CFStringRef) key;
490 CFDataRef message = (CFDataRef) value;
491 bool rx = sendToPeer(transport, (__bridge CFStringRef)transport->circleName, peerID, message, error);
499 -(CFDictionaryRef)CF_RETURNS_RETAINED SOSTransportMessageHandlePeerMessageReturnsHandledCopy:(SOSMessageIDSTest*) transport peerMessages:(CFMutableDictionaryRef) circle_peer_messages_table err:(CFErrorRef *)error{
500 CFMutableDictionaryRef handled = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
501 CFDictionaryRef peerToMessage = CFDictionaryGetValue(circle_peer_messages_table, (__bridge CFStringRef)transport.circleName);
502 CFMutableArrayRef handled_peers = CFArrayCreateMutableForCFTypes(kCFAllocatorDefault);
503 CFDictionaryAddValue(handled, (__bridge CFStringRef)transport.circleName, handled_peers);
506 CFDictionaryForEach(peerToMessage, ^(const void *key, const void *value) {
507 CFStringRef peer_id = (CFStringRef) key;
508 CFDataRef peer_message = (CFDataRef) value;
509 CFErrorRef localError = NULL;
511 if ([transport SOSTransportMessageHandlePeerMessage:transport id:peer_id cm:peer_message err:&localError]){
512 CFArrayAppendValue(handled_peers, key);
514 secdebug("transport", "%@ KVSTransport handle message failed: %@", peer_id, localError);
516 CFReleaseNull(localError);
519 CFReleaseNull(handled_peers);
524 -(bool) SOSTransportMessageFlushChanges:(SOSMessageKVSTest*) transport err:(CFErrorRef *)error
529 SOSAccount* SOSTransportMessageKVSTestGetAccount(SOSMessageKVSTest* transport) {
530 return transport.account;
535 //MARK IDS Message Test Transport
538 @implementation SOSMessageIDSTest
539 -(SOSMessageIDSTest*) initWithAccount:(SOSAccount*)acct andAccountName:(CFStringRef) aN andCircleName:(CFStringRef) cN err:(CFErrorRef *)error
543 self.engine = SOSDataSourceFactoryGetEngineForDataSourceName(acct.factory, cN, NULL);
544 self.useFragmentation = kCFBooleanTrue;
546 self.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
547 self.circleName = (__bridge NSString*)cN;
548 self.accountName = CFRetainSafe(aN);
550 if(!message_transports)
551 message_transports = CFArrayCreateMutable(kCFAllocatorDefault, 0, NULL);
552 CFArrayAppendValue(message_transports, (__bridge CFTypeRef)self);
553 SOSRegisterTransportMessage((SOSMessageIDSTest*)self);
559 -(bool) SOSTransportMessageIDSGetIDSDeviceID:(SOSAccount*)acct{
563 -(CFIndex) SOSTransportMessageGetTransportType
567 -(CFStringRef) SOSTransportMessageGetCircleName
569 return (__bridge CFStringRef)circleName;
571 -(CFTypeRef) SOSTransportMessageGetEngine
575 -(SOSAccount*) SOSTransportMessageGetAccount
580 void SOSTransportMessageIDSTestSetName(SOSMessageIDSTest* transport, CFStringRef acctName){
581 transport.accountName = acctName;
585 static HandleIDSMessageReason checkMessageValidity(SOSAccount* account, CFStringRef fromDeviceID, CFStringRef fromPeerID, CFStringRef *peerID, SOSPeerInfoRef *theirPeerInfo){
586 SOSAccountTrustClassic *trust = account.trust;
587 __block HandleIDSMessageReason reason = kHandleIDSMessageDontHandle;
589 SOSCircleForEachPeer(trust.trustedCircle, ^(SOSPeerInfoRef peer) {
590 CFStringRef deviceID = SOSPeerInfoCopyDeviceID(peer);
591 CFStringRef pID = SOSPeerInfoGetPeerID(peer);
593 if( deviceID && pID && fromPeerID && fromDeviceID && CFStringGetLength(fromPeerID) != 0 ){
594 if(CFStringCompare(pID, fromPeerID, 0) == 0){
595 if(CFStringGetLength(deviceID) == 0){
596 secnotice("ids transport", "device ID was empty in the peer list, holding on to message");
597 CFReleaseNull(deviceID);
598 reason = kHandleIDSMessageNotReady;
601 else if(CFStringCompare(fromDeviceID, deviceID, 0) != 0){ //IDSids do not match, ghost
602 reason = kHandleIDSmessageDeviceIDMismatch;
603 CFReleaseNull(deviceID);
606 else if(CFStringCompare(deviceID, fromDeviceID, 0) == 0){
608 *theirPeerInfo = peer;
609 CFReleaseNull(deviceID);
610 reason = kHandleIDSMessageSuccess;
615 CFReleaseNull(deviceID);
620 const char *kMessageKeyIDSDataMessage = "idsDataMessage";
621 const char *kMessageKeyDeviceID = "deviceID";
622 const char *kMessageKeyPeerID = "peerID";
623 const char *kMessageKeySendersPeerID = "sendersPeerID";
625 -(HandleIDSMessageReason) SOSTransportMessageIDSHandleMessage:(SOSAccount*)acct m:(CFDictionaryRef) message err:(CFErrorRef *)error
627 secnotice("IDS Transport", "SOSTransportMessageIDSHandleMessage!");
629 CFStringRef dataKey = CFStringCreateWithCString(kCFAllocatorDefault, kMessageKeyIDSDataMessage, kCFStringEncodingASCII);
630 CFStringRef deviceIDKey = CFStringCreateWithCString(kCFAllocatorDefault, kMessageKeyDeviceID, kCFStringEncodingASCII);
631 CFStringRef sendersPeerIDKey = CFStringCreateWithCString(kCFAllocatorDefault, kMessageKeySendersPeerID, kCFStringEncodingASCII);
632 CFStringRef ourPeerIdKey = CFStringCreateWithCString(kCFAllocatorDefault, kMessageKeyPeerID, kCFStringEncodingASCII);
634 HandleIDSMessageReason result = kHandleIDSMessageSuccess;
636 CFDataRef messageData = asData(CFDictionaryGetValue(message, dataKey), NULL);
637 __block CFStringRef fromDeviceID = asString(CFDictionaryGetValue(message, deviceIDKey), NULL);
638 __block CFStringRef fromPeerID = (CFStringRef)CFDictionaryGetValue(message, sendersPeerIDKey);
639 CFStringRef ourPeerID = asString(CFDictionaryGetValue(message, ourPeerIdKey), NULL);
641 CFStringRef peerID = NULL;
642 SOSPeerInfoRef theirPeer = NULL;
644 require_action_quiet(fromDeviceID, exit, result = kHandleIDSMessageDontHandle);
645 require_action_quiet(fromPeerID, exit, result = kHandleIDSMessageDontHandle);
646 require_action_quiet(messageData && CFDataGetLength(messageData) != 0, exit, result = kHandleIDSMessageDontHandle);
647 require_action_quiet(SOSAccountHasFullPeerInfo(account, error), exit, result = kHandleIDSMessageNotReady);
648 require_action_quiet(ourPeerID && [account.peerID isEqual: (__bridge NSString*) ourPeerID], exit, result = kHandleIDSMessageDontHandle; secnotice("IDS Transport","ignoring message for: %@", ourPeerID));
650 require_quiet((result = checkMessageValidity( account, fromDeviceID, fromPeerID, &peerID, &theirPeer)) == kHandleIDSMessageSuccess, exit);
652 if ([account.ids_message_transport SOSTransportMessageHandlePeerMessage:account.ids_message_transport id:peerID cm:messageData err:error]) {
653 CFMutableDictionaryRef peersToSyncWith = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
654 CFMutableSetRef peerIDs = CFSetCreateMutableForCFTypes(kCFAllocatorDefault);
655 CFSetAddValue(peerIDs, peerID);
656 SOSAccountTrustClassic* trust = account.trust;
657 //sync using fragmentation?
658 if(SOSPeerInfoShouldUseIDSMessageFragmentation(trust.peerInfo, theirPeer)){
659 //set useFragmentation bit
660 [account.ids_message_transport SOSTransportMessageIDSSetFragmentationPreference:account.ids_message_transport pref: kCFBooleanTrue];
663 [account.ids_message_transport SOSTransportMessageIDSSetFragmentationPreference:account.ids_message_transport pref: kCFBooleanFalse];
666 if(![account.ids_message_transport SOSTransportMessageSyncWithPeers:account.ids_message_transport p:peerIDs err:error]){
667 secerror("SOSTransportMessageIDSHandleMessage Could not sync with all peers: %@", *error);
669 secnotice("IDS Transport", "Synced with all peers!");
672 CFReleaseNull(peersToSyncWith);
673 CFReleaseNull(peerIDs);
675 if(error && *error != NULL){
676 CFStringRef errorMessage = CFErrorCopyDescription(*error);
677 if (-25308 == CFErrorGetCode(*error)) { // tell KeychainSyncingOverIDSProxy to call us back when device unlocks
678 result = kHandleIDSMessageLocked;
679 }else{ //else drop it, couldn't handle the message
680 result = kHandleIDSMessageDontHandle;
682 secerror("IDS Transport Could not handle message: %@, %@", messageData, *error);
683 CFReleaseNull(errorMessage);
686 else{ //no error but failed? drop it, log message
687 secerror("IDS Transport Could not handle message: %@", messageData);
688 result = kHandleIDSMessageDontHandle;
694 CFReleaseNull(ourPeerIdKey);
695 CFReleaseNull(sendersPeerIDKey);
696 CFReleaseNull(deviceIDKey);
697 CFReleaseNull(dataKey);
701 -(CFDictionaryRef)CF_RETURNS_RETAINED SOSTransportMessageHandlePeerMessageReturnsHandledCopy:(SOSMessageIDSTest*) transport peerMessages:(CFMutableDictionaryRef)message err:(CFErrorRef *)error
704 CFMutableDictionaryRef handled = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
705 CFDictionaryRef peerToMessage = CFDictionaryGetValue(message, (__bridge CFTypeRef)(transport.circleName));
706 CFMutableArrayRef handled_peers = CFArrayCreateMutableForCFTypes(kCFAllocatorDefault);
708 secerror("Received IDS message!");
710 CFDictionaryForEach(peerToMessage, ^(const void *key, const void *value) {
711 CFStringRef peer_id = asString(key, NULL);
712 CFDataRef peer_message = asData(value, NULL);
713 CFErrorRef localError = NULL;
715 if (peer_id && peer_message && [transport SOSTransportMessageHandlePeerMessage:transport id:peer_id cm:peer_message err:&localError]) {
716 CFArrayAppendValue(handled_peers, key);
718 secnotice("transport", "%@ KVSTransport handle message failed: %@", peer_id, localError);
720 CFReleaseNull(localError);
723 CFDictionaryAddValue(handled, (__bridge CFStringRef)(transport.circleName), handled_peers);
724 CFReleaseNull(handled_peers);
729 static void SOSTransportMessageIDSTestAddBulkToChanges(SOSMessageIDSTest* transport, CFDictionaryRef updates){
730 #ifndef __clang_analyzer__ // The analyzer thinks transport.changes is a leak, but I don't know why.
731 if (transport.changes == NULL) {
732 transport.changes = CFDictionaryCreateMutableCopy(kCFAllocatorDefault, CFDictionaryGetCount(updates), updates);
736 CFDictionaryForEach(updates, ^(const void *key, const void *value) {
737 CFDictionaryAddValue(transport.changes, key, value);
742 static bool sendDataToPeerIDSTest(SOSMessageIDSTest* transport, CFStringRef circleName, CFStringRef deviceID, CFStringRef peerID, CFDataRef message, CFErrorRef *error)
744 secerror("sending message through test transport: %@", message);
745 NSString* myID = transport.account.peerID;
746 CFStringRef message_to_peer_key = SOSMessageKeyCreateFromTransportToPeer(transport, (__bridge CFStringRef) myID, peerID);
747 CFDictionaryRef a_message_to_a_peer = CFDictionaryCreateForCFTypes(NULL, message_to_peer_key, message, NULL);
749 SOSTransportMessageIDSTestAddBulkToChanges(transport, a_message_to_a_peer);
751 CFReleaseNull(message_to_peer_key);
752 CFReleaseNull(a_message_to_a_peer);
756 static bool sendDictionaryToPeerIDSTest(SOSMessageIDSTest* transport, CFStringRef circleName, CFStringRef deviceID, CFStringRef peerID, CFDictionaryRef message, CFErrorRef *error)
758 secerror("sending message through test transport: %@", message);
759 NSString* myID = transport.account.peerID;
760 CFStringRef message_to_peer_key = SOSMessageKeyCreateFromTransportToPeer(transport, (__bridge CFStringRef) myID, peerID);
761 CFDictionaryRef a_message_to_a_peer = CFDictionaryCreateForCFTypes(NULL, message_to_peer_key, message, NULL);
763 SOSTransportMessageIDSTestAddBulkToChanges(transport, a_message_to_a_peer);
765 CFReleaseNull(message_to_peer_key);
766 CFReleaseNull(a_message_to_a_peer);
771 -(bool) SOSTransportMessageSyncWithPeers:(SOSMessageIDSTest*) transport p:(CFSetRef)peers err:(CFErrorRef *)error
773 // Each entry is keyed by circle name and contains a list of peerIDs
774 __block bool result = true;
776 CFSetForEach(peers, ^(const void *value) {
777 CFStringRef peerID = asString(value, NULL);
779 secnotice("transport", "IDS sync with peerIDs %@", peerID);
780 result &= [transport SOSTransportMessageSendMessageIfNeeded:transport id:(__bridge CFStringRef)transport.circleName pID:peerID err:error];
787 -(bool) SOSTransportMessageSendMessages:(SOSMessageIDSTest*) transport pm:(CFDictionaryRef) peer_messages err:(CFErrorRef *)error
789 __block bool result = true;
790 CFDictionaryForEach(peer_messages, ^(const void *key, const void *value) {
791 CFStringRef idsDeviceID = NULL;;
792 CFStringRef peerID = asString(key, NULL);
793 SOSPeerInfoRef pi = NULL;
795 if(!CFEqualSafe(peerID, (__bridge CFStringRef) transport.account.peerID)){
797 pi = SOSAccountCopyPeerWithID(transport.account, peerID, NULL);
799 idsDeviceID = SOSPeerInfoCopyDeviceID(pi);
800 if(idsDeviceID != NULL){
802 CFDictionaryRef messageDictionary = asDictionary(value, NULL);
803 if (messageDictionary) {
804 result &= sendDictionaryToPeerIDSTest(transport, (__bridge CFStringRef)transport.circleName, idsDeviceID, peerID, messageDictionary, error);
806 CFDataRef messageData = asData(value, NULL);
808 result &= sendDataToPeerIDSTest(transport, (__bridge CFStringRef)transport.circleName, idsDeviceID, peerID, messageData, error);
815 CFReleaseNull(idsDeviceID);
822 -(bool) SOSTransportMessageFlushChanges:(SOSMessageIDSTest*) transport err:(CFErrorRef *)error
827 -(bool) SOSTransportMessageCleanupAfterPeerMessages:(SOSMessageIDSTest*) transport peers:(CFDictionaryRef) peers err:(CFErrorRef*) error
832 void SOSTransportMessageIDSTestClearChanges(SOSMessageIDSTest* transport)
834 transport.changes = CFDictionaryCreateMutableForCFTypes(kCFAllocatorDefault);
837 CFMutableDictionaryRef SOSTransportMessageIDSTestGetChanges(SOSMessageIDSTest* transport)
839 return transport.changes;
842 SOSAccount* SOSTransportMessageIDSTestGetAccount(SOSMessageIDSTest* transport)
844 return transport.account;
847 CFStringRef SOSTransportMessageIDSTestGetName(SOSMessageIDSTest* transport){
848 return transport.accountName;
854 void SOSAccountUpdateTestTransports(SOSAccount* account, CFDictionaryRef gestalt){
855 CFStringRef new_name = (CFStringRef)CFDictionaryGetValue(gestalt, kPIUserDefinedDeviceNameKey);
857 SOSTransportKeyParameterTestSetName((CKKeyParameterTest*)account.key_transport, new_name);
858 SOSTransportCircleTestSetName((SOSCircleStorageTransportTest*)account.circle_transport, new_name);
859 SOSTransportMessageKVSTestSetName((SOSMessageKVSTest*)account.kvs_message_transport, new_name);
860 SOSTransportMessageIDSTestSetName((SOSMessageIDSTest*)account.ids_message_transport, new_name);
864 static CF_RETURNS_RETAINED SOSCircleRef SOSAccountEnsureCircleTest(SOSAccount* a, CFStringRef name, CFStringRef accountName)
866 CFErrorRef localError = NULL;
867 SOSAccountTrustClassic *trust = a.trust;
869 SOSCircleRef circle = CFRetainSafe([a.trust getCircle:&localError]);
870 if(!circle || isSOSErrorCoded(localError, kSOSErrorIncompatibleCircle)){
871 secnotice("circle", "Error retrieving the circle: %@", localError);
872 CFReleaseNull(localError);
873 CFReleaseNull(circle);
875 circle = SOSCircleCreate(kCFAllocatorDefault, name, &localError);
877 [trust setTrustedCircle:circle];
880 secnotice("circle", "Could not create circle: %@", localError);
882 CFReleaseNull(localError);
885 if(![trust ensureFullPeerAvailable:(__bridge CFDictionaryRef)(a.gestalt) deviceID:(__bridge CFStringRef)(a.deviceID) backupKey:(__bridge CFDataRef)(a.backup_key) err:&localError])
887 secnotice("circle", "had an error building full peer: %@", localError);
888 CFReleaseNull(localError);
892 CFReleaseNull(localError);
896 bool SOSAccountEnsureFactoryCirclesTest(SOSAccount* a, CFStringRef accountName)
903 CFStringRef circle_name = SOSDataSourceFactoryCopyName(a.factory);
906 CFReleaseSafe(SOSAccountEnsureCircleTest(a, (CFStringRef)circle_name, accountName));
908 CFReleaseNull(circle_name);
914 bool SOSAccountInflateTestTransportsForCircle(SOSAccount* account, CFStringRef circleName, CFStringRef accountName, CFErrorRef *error){
915 bool success = false;
917 if(account.key_transport == nil){
918 account.key_transport = (CKKeyParameter*)[[CKKeyParameterTest alloc] initWithAccount:account andName:accountName andCircleName:circleName];
919 require_quiet(account.key_transport, fail);
922 if(account.circle_transport == nil){
923 account.circle_transport = (SOSKVSCircleStorageTransport*)[[SOSCircleStorageTransportTest alloc] initWithAccount:account andWithAccountName:accountName andCircleName:circleName];
924 require_quiet(account.circle_transport, fail);
926 if(account.kvs_message_transport == nil){
927 account.kvs_message_transport = (SOSMessageKVS*)[[SOSMessageKVSTest alloc] initWithAccount:account andName:accountName andCircleName:circleName];
929 if(account.ids_message_transport == nil){
930 account.ids_message_transport = (SOSMessageIDS*)[[SOSMessageIDSTest alloc] initWithAccount:account andAccountName:accountName andCircleName:circleName err:NULL];