]> git.saurik.com Git - apple/security.git/blobdiff - Security/sec/SOSCircle/Regressions/sc-70-engine.c
Security-57031.1.35.tar.gz
[apple/security.git] / Security / sec / SOSCircle / Regressions / sc-70-engine.c
diff --git a/Security/sec/SOSCircle/Regressions/sc-70-engine.c b/Security/sec/SOSCircle/Regressions/sc-70-engine.c
new file mode 100644 (file)
index 0000000..59991a9
--- /dev/null
@@ -0,0 +1,391 @@
+/*
+ * Copyright (c) 2012-2014 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 <SecureObjectSync/SOSEngine.h>
+#include <SecureObjectSync/SOSPeer.h>
+
+#include "SOSCircle_regressions.h"
+
+#include <corecrypto/ccsha2.h>
+#include <Security/SecBase64.h>
+
+#include <utilities/SecCFWrappers.h>
+
+#include <stdint.h>
+#include "SOSTestDataSource.h"
+
+#if 1
+static int kTestTestCount = 0;
+#else
+static int kTestTestCount = 61;
+
+static void tests(void)
+{
+    CFErrorRef error = NULL;
+
+    /* Transport. */
+    __block unsigned msg_count = 0;
+    uint8_t msg_digest_buffer[CCSHA256_OUTPUT_SIZE];
+    uint8_t *msg_digest = msg_digest_buffer;
+    
+    SOSPeerSendBlock sendBlock = ^bool (CFDataRef message, CFErrorRef *error) {
+        size_t msglen = CFDataGetLength(message);
+        const uint8_t *msg = CFDataGetBytePtr(message);
+        const struct ccdigest_info *sha256 = ccsha256_di();
+        if (msg_count++ == 0) {
+            /* message n=0 */
+            ccdigest(sha256, msglen, msg, msg_digest);
+        } else {
+            /* message n=n+1 */
+            ccdigest_di_decl(sha256, sha256_ctx);
+            ccdigest_init(sha256, sha256_ctx);
+            ccdigest_update(sha256, sha256_ctx, sizeof(msg_digest_buffer), msg_digest);
+            ccdigest_update(sha256, sha256_ctx, msglen, msg);
+            ccdigest_final(sha256, sha256_ctx, msg_digest);
+        }
+        size_t encmaxlen = SecBase64Encode(msg, msglen, NULL, 0);
+        CFMutableDataRef encoded = CFDataCreateMutable(NULL, encmaxlen);
+        CFDataSetLength(encoded, encmaxlen);
+        SecBase64Result rc;
+        char *enc = (char *)CFDataGetMutableBytePtr(encoded);
+#ifndef NDEBUG
+        size_t enclen =
+#endif
+            SecBase64Encode2(msg, msglen,
+                             enc,
+                             encmaxlen, kSecB64_F_LINE_LEN_USE_PARAM,
+                             64, &rc);
+        assert(enclen < INT32_MAX);
+//      printf("=== BEGIN SOSMESSAGE ===\n%.*s\n=== END SOSMESSAGE ===\n", (int)enclen, enc);
+        CFRelease(encoded);
+        return true;
+    };
+
+    /* Create peer test. */
+    CFStringRef peer_id = CFSTR("peer 70");
+    SOSPeerRef peer;
+    ok(peer = SOSPeerCreateSimple(peer_id, kSOSPeerVersion, &error),
+       "create peer: %@", error);
+    CFReleaseNull(error);
+
+    /* DataSource */
+    SOSDataSourceRef ds = SOSTestDataSourceCreate();
+
+    /* Create engine test. */
+    SOSEngineRef engine;
+    ok(engine = SOSEngineCreate(ds, &error), "create engine: %@", error);
+    CFReleaseNull(error);
+
+    /* Make sure the engine did not yet send a any messages to the peer. */
+    unsigned expected_msg_count = 0;
+    is(msg_count, expected_msg_count, "Engine sent %d/%d messages",
+       msg_count, expected_msg_count);
+
+    /* Test passing peer messages to the engine. */
+    CFDataRef message;
+
+    /* Hand an empty message to the engine for handeling. */
+    message = CFDataCreate(NULL, NULL, 0);
+    ok(false == SOSEngineHandleMessage(engine, peer, message, &error),
+       "handle empty message: %@", error);
+    ok(false == SOSPeerSendMessageIfNeeded(engine, peer, &error, sendBlock),
+       "send response: %@", error);
+    CFReleaseNull(error);
+    CFReleaseNull(message);
+    /* Make sure the engine did not yet send a response to the peer. */
+    is(msg_count, expected_msg_count, "Engine sent %d/%d messages",
+       msg_count, expected_msg_count);
+
+
+    /* Create manifest digest message. */
+    SKIP: {
+        skip("Create manifest digest message failed", 2,
+             ok(message = SOSEngineCreateManifestDigestMessage(engine, peer,
+                                                               &error),
+                "create manifest digest message: %@", error));
+        CFReleaseNull(error);
+        /* Pass manifest digest message to the engine for handeling. */
+        skip("", 1,
+             ok(SOSEngineHandleMessage(engine, peer, message, &error),
+                "handle manifest digest message: %@", error));
+        /* Make sure the engine sent a response to the peer. */
+        expected_msg_count++;
+        is(msg_count, expected_msg_count, "Engine sent %d/%d messages",
+           msg_count, expected_msg_count);
+    }
+    CFReleaseNull(message);
+    CFReleaseNull(error);
+
+    /* Create manifest message. */
+    SKIP: {
+        skip("Create manifest message failed", 2,
+             ok(message = SOSEngineCreateManifestMessage(engine, peer, &error),
+             "create manifest message: %@", error));
+        CFReleaseNull(error);
+        /* Pass manifest message to the engine for handeling. */
+        skip("", 1,
+             ok(SOSEngineHandleMessage(engine, peer, message, &error),
+                "handle manifest message: %@", error));
+        /* Make sure the engine sent a response to the peer. */
+        expected_msg_count++;
+        is(msg_count, expected_msg_count, "Engine sent %d/%d messages",
+           msg_count, expected_msg_count);
+    }
+    CFReleaseNull(message);
+    CFReleaseNull(error);
+
+    /* Create manifest and objects message. */
+    SKIP: {
+        skip("Create manifest and objects message failed", 2,
+             ok(message = SOSEngineCreateManifestAndObjectsMessage(engine, peer,
+                                                                   &error),
+                "create manifest and objects message: %@", error));
+        CFReleaseNull(error);
+        /* Pass manifest and objects message to the engine for handeling. */
+        skip("", 1,
+             ok(SOSEngineHandleMessage(engine, peer, message, &error),
+                "handle manifest and objects message: %@", error));
+        /* Make sure the engine sent a response to the peer. */
+        expected_msg_count++;
+        is(msg_count, expected_msg_count, "Engine sent %d/%d messages",
+           msg_count, expected_msg_count);
+    }
+    CFReleaseNull(message);
+    CFReleaseNull(error);
+
+    /* Clean up. */
+    CFReleaseSafe(peer);
+    SOSEngineDispose(engine);
+}
+
+static CFStringRef SOSMessageCopyDigestHex(CFDataRef message) {
+    uint8_t digest[CCSHA1_OUTPUT_SIZE];
+    ccdigest(ccsha1_di(), CFDataGetLength(message), CFDataGetBytePtr(message), digest);
+    CFMutableStringRef hex = CFStringCreateMutable(0, 2 * sizeof(digest));
+    for (unsigned int ix = 0; ix < sizeof(digest); ++ix) {
+        CFStringAppendFormat(hex, 0, CFSTR("%02X"), digest[ix]);
+    }
+    return hex;
+}
+
+static void testsync(const char *name, void (^aliceInit)(SOSDataSourceRef ds), void (^bobInit)(SOSDataSourceRef ds), CFStringRef msg, ...) {
+    CFErrorRef error = NULL;
+
+    /* Setup Alice engine, dataSource and peer for Alice to talk to Bob */
+    SOSDataSourceRef aliceDataSource = SOSTestDataSourceCreate();
+    SOSEngineRef aliceEngine;
+    ok(aliceEngine = SOSEngineCreate(aliceDataSource, &error), "create alice engine: %@", error);
+    CFReleaseNull(error);
+    CFStringRef bobID = CFStringCreateWithFormat(kCFAllocatorDefault, 0, CFSTR("Bob-%s"), name);
+    
+    __block CFDataRef queued_message = NULL;
+    
+    SOSPeerSendBlock enqueueMessage =  ^bool (CFDataRef message, CFErrorRef *error) {
+        if (queued_message)
+            fail("We already had an unproccessed message");
+        
+        queued_message = (CFDataRef) CFRetain(message);
+        return true;
+    };
+
+    CFDataRef (^dequeueMessage)() = ^CFDataRef () {        
+        CFDataRef result = queued_message;
+        queued_message = NULL;
+        
+        return result;
+    };
+    
+    SOSPeerRef bobPeer;
+    ok(bobPeer = SOSPeerCreateSimple(bobID, kSOSPeerVersion, &error, enqueueMessage),
+       "create peer: %@", error);
+
+    /* Setup Bob engine, dataSource and peer for Bob to talk to Alice */
+    SOSDataSourceRef bobDataSource = SOSTestDataSourceCreate();
+    SOSEngineRef bobEngine;
+    ok(bobEngine = SOSEngineCreate(bobDataSource, &error), "create bob engine: %@", error);
+    CFReleaseNull(error);
+    CFStringRef aliceID = CFStringCreateWithFormat(kCFAllocatorDefault, 0, CFSTR("Alice-%s"), name);
+
+    SOSPeerRef alicePeer;
+    ok(alicePeer = SOSPeerCreateSimple(aliceID, kSOSPeerVersion, &error, enqueueMessage),
+       "create peer: %@", error);
+    CFReleaseNull(error);
+
+    /* Now call provided setup blocks to populate the dataSources with
+       interesting stuff. */
+    aliceInit(aliceDataSource);
+    bobInit(bobDataSource);
+
+    CFDataRef message;
+
+       va_list msgs;
+       va_start(msgs, msg);
+
+    int msg_index = 0;
+    bool alice = false;
+    for (;;) {
+        message = dequeueMessage();
+        msg_index++;
+        /* We are expecting a message and msg is it's digest. */
+        if (message) {
+            CFStringRef messageDigestStr = SOSMessageCopyDigestHex(message);
+            if (msg) {
+                bool handeled = SOSEngineHandleMessage(alice ? aliceEngine : bobEngine, alice ? bobPeer : alicePeer, message, &error);
+                if (!CFEqual(messageDigestStr, msg)) {
+                    if (handeled) {
+                        fail("%s %s received message [%d] digest %@ != %@ %@", name, alice ? "Alice" : "Bob", msg_index, messageDigestStr, msg, message);
+                    } else {
+                        fail("%s %s failed to handle message [%d] digest %@ != %@ %@: %@", name, alice ? "Alice" : "Bob", msg_index, messageDigestStr, msg, message, error);
+                        CFReleaseNull(error);
+                    }
+                } else if (handeled) {
+                    pass("%s %s handled message [%d] %@", name, alice ? "Alice" : "Bob", msg_index, message);
+                } else {
+                    fail("%s %s failed to handle message [%d] %@: %@", name, alice ? "Alice" : "Bob", msg_index, message, error);
+                    CFReleaseNull(error);
+                }
+            } else {
+                fail("%s %s sent extra message [%d] with digest %@: %@", name, alice ? "Bob" : "Alice", msg_index, messageDigestStr, message);
+            }
+            CFRelease(messageDigestStr);
+            CFRelease(message);
+        } else {
+            if (msg) {
+                fail("%s %s expected message [%d] with digest %@, none received", name, alice ? "Alice" : "Bob", msg_index, msg);
+            } else {
+                /* Compare alice and bobs dataSources databases. */
+                ok(CFEqual(SOSTestDataSourceGetDatabase(aliceDataSource), SOSTestDataSourceGetDatabase(bobDataSource)), "%s Alice and Bob are in sync", name);
+            }
+        }
+
+        if (msg) {
+            alice = !alice;
+            msg = va_arg(msgs, CFStringRef);
+        } else
+            break;
+    }
+
+       va_end(msgs);
+
+       
+    
+    
+    SOSEngineDispose(aliceEngine); // Also disposes aliceDataSource
+    CFReleaseSafe(alicePeer);
+    CFReleaseSafe(aliceID);
+
+    SOSEngineDispose(bobEngine); // Also disposes bobDataSource
+    CFReleaseSafe(bobPeer);
+    CFReleaseSafe(bobID);
+}
+
+#if 0
+static void synctests(void) {
+    // Sync between 2 empty dataSources
+    testsync("empty",
+             ^ (SOSDataSourceRef dataSource) {},
+             ^ (SOSDataSourceRef dataSource) {},
+             CFSTR("2AF312E092D67308A0083DFFBF2B6B754B967864"),
+             CFSTR("2AF312E092D67308A0083DFFBF2B6B754B967864"),
+             CFSTR("2AF312E092D67308A0083DFFBF2B6B754B967864"),
+             NULL);
+
+    // Sync a dataSource with one object to an empty dataSource
+    testsync("alice1",
+             ^ (SOSDataSourceRef dataSource) {
+                 CFErrorRef error = NULL;
+                 SOSObjectRef object = SOSDataSourceCreateGenericItem(dataSource, CFSTR("test_account"), CFSTR("test service"));
+                 ok(dataSource->add(dataSource, object, &error), "dataSource added object %@", error);
+                 CFReleaseSafe(object);
+                 CFReleaseNull(error);
+             },
+             ^ (SOSDataSourceRef dataSource) {},
+             CFSTR("ADAA3ACE75ED516CB91893413EE9CC9ED04CA47B"),
+             CFSTR("147B6C509908CC4A9FC4263973A842104A64CE01"),
+             CFSTR("019B494F3C06B48BB02C280AF1E19AD861A7003C"),
+             CFSTR("ADAA3ACE75ED516CB91893413EE9CC9ED04CA47B"),
+             NULL);
+
+    // Sync a dataSource with one object to another dataSource with the same object
+    testsync("alice1bob1",
+             ^ (SOSDataSourceRef dataSource) {
+                 CFErrorRef error = NULL;
+                 SOSObjectRef object = SOSDataSourceCreateGenericItem(dataSource, CFSTR("test_account"), CFSTR("test service"));
+                 ok(dataSource->add(dataSource, object, &error), "dataSource added object %@", error);
+                 CFReleaseSafe(object);
+                 CFReleaseNull(error);
+             },
+             ^ (SOSDataSourceRef dataSource) {
+                 CFErrorRef error = NULL;
+                 SOSObjectRef object = SOSDataSourceCreateGenericItem(dataSource, CFSTR("test_account"), CFSTR("test service"));
+                 ok(dataSource->add(dataSource, object, &error), "dataSource added object %@", error);
+                 CFReleaseSafe(object);
+                 CFReleaseNull(error);
+             },
+             CFSTR("ADAA3ACE75ED516CB91893413EE9CC9ED04CA47B"),
+             CFSTR("ADAA3ACE75ED516CB91893413EE9CC9ED04CA47B"),
+             CFSTR("ADAA3ACE75ED516CB91893413EE9CC9ED04CA47B"),
+             NULL);
+
+    // Sync a dataSource with one object to another dataSource with the same object
+    testsync("alice1bob2",
+             ^ (SOSDataSourceRef dataSource) {
+                 CFErrorRef error = NULL;
+                 SOSObjectRef object = SOSDataSourceCreateGenericItem(dataSource, CFSTR("test_account"), CFSTR("test service"));
+                 ok(dataSource->add(dataSource, object, &error), "dataSource added object %@", error);
+                 CFReleaseSafe(object);
+                 CFReleaseNull(error);
+             },
+             ^ (SOSDataSourceRef dataSource) {
+                 CFErrorRef error = NULL;
+                 SOSObjectRef object = SOSDataSourceCreateGenericItem(dataSource, CFSTR("test_account"), CFSTR("test service"));
+                 ok(dataSource->add(dataSource, object, &error), "dataSource added object %@", error);
+                 CFReleaseSafe(object);
+                 object = SOSDataSourceCreateGenericItem(dataSource, CFSTR("account1"), CFSTR("service1"));
+                 ok(dataSource->add(dataSource, object, &error), "dataSource added object %@", error);
+                 CFReleaseSafe(object);
+                 CFReleaseNull(error);
+             },
+             CFSTR("ADAA3ACE75ED516CB91893413EE9CC9ED04CA47B"),
+             CFSTR("D4049A1063CFBF7CAF8424E13DE3CE926FF5856C"),
+             CFSTR("9624EA855BBED6B668868BB723443E804D04F6A1"),
+             CFSTR("063E097CCD4FEB7F3610ED12B3DA828467314846"),
+             CFSTR("D1B3944E3084425F41B2C2EA0BE82170E10AA37D"),
+             NULL);
+
+}
+#endif
+
+#endif
+
+int sc_70_engine(int argc, char *const *argv)
+{
+    plan_tests(kTestTestCount);
+
+    //tests();
+    //synctests();
+    //testsync(NULL, NULL, NULL, NULL);
+
+       return 0;
+}