为 MCSession 授权证书

Posted

技术标签:

【中文标题】为 MCSession 授权证书【英文标题】:Authorizing a certificate for MCSession 【发布时间】:2014-04-23 19:10:26 【问题描述】:

我在钥匙串中创建了一个自签名证书,并且我正在使用 MultipeerConnectivity 框架在设备之间传输数据。

当我们创建会话时,我们可以在 MCSession 中邀请对等方时传递 securityIdentity。

  - (instancetype)initWithPeer:(MCPeerID *)myPeerID securityIdentity:(NSArray *)identity encryptionPreference:(MCEncryptionPreference)encryptionPreference;

但是 securityIdentity 是 NSArray,我们如何在 NSArray 中传递证书,如何在 NSArray 中进行身份验证

- (void)session:(MCSession *)session didReceiveCertificate:(NSArray *)certificate fromPeer:(MCPeerID *)peerID certificateHandler:(void(^)(BOOL accept))certificateHandler;

【问题讨论】:

【参考方案1】:

它是一个包含信息的数组,可用于将本地对等点识别为其他附近对等点。

该数组包含有助于向其他人识别对等方的对象。 第一个是具有 SecKeyRef 对象和相关 SecCertificateRef 对象的 SecIdentityRef 对象。 (类似于苹果开发者证书和私钥对)

数组中的其他元素可以是 SecCertificateRef 对象,代表验证 SecIdentityRef 可能需要的中间证书。

接收端必须验证 SecIdentityRef 所代表的身份。

下面是从p12文件中获取SecIdentityRef的代码

- (SecIdentityRef)getClientCertificate 

    SecIdentityRef identity = nil;
    NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    NSString *documentsDirectoryPath = [paths objectAtIndex:0];
    NSString *myFilePath = [documentsDirectoryPath stringByAppendingPathComponent:@"cert_key_pair.p12"]; 
    NSData *PKCS12Data = [NSData dataWithContentsOfFile:myFilePath];

    CFDataRef inPKCS12Data = (__bridge CFDataRef)PKCS12Data;
    CFStringRef password = CFSTR("password");
    const void *keys[] =  kSecImportExportPassphrase ;//kSecImportExportPassphrase ;
    const void *values[] =  password ;
    CFDictionaryRef options = CFDictionaryCreate(NULL, keys, values, 1, NULL, NULL);
    CFArrayRef items = CFArrayCreate(NULL, 0, 0, NULL);
    OSStatus securityError = SecPKCS12Import(inPKCS12Data, options, &items);
    CFRelease(options);
    CFRelease(password);
    if (securityError == errSecSuccess) 
        NSLog(@"Success opening p12 certificate. Items: %ld", CFArrayGetCount(items));
        CFDictionaryRef identityDict = CFArrayGetValueAtIndex(items, 0);
        identity = (SecIdentityRef)CFDictionaryGetValue(identityDict, kSecImportItemIdentity);
     else 
        NSLog(@"Error opening Certificate.");
    

    return identity;

获取策略引用对象并评估信任

- (void)session:(MCSession *)session didReceiveCertificate:(NSArray *)certificate fromPeer:     (MCPeerID *)peerID certificateHandler:(void (^)(BOOL accept))certificateHandler
 
    SecCertificateRef myCert;
    myCert = [certificate objectAtIndex:0];    // 1

    SecPolicyRef myPolicy = SecPolicyCreateBasicX509();         // 2

    SecCertificateRef certArray[1] =  myCert ;
    CFArrayRef myCerts = CFArrayCreate(
                                   NULL, (void *)certArray,
                                   1, NULL);
    SecTrustRef myTrust;
    OSStatus status = SecTrustCreateWithCertificates(
                                                myCerts,
                                                myPolicy,
                                                &myTrust);  // 3

    SecTrustResultType trustResult;
    if (status == noErr) 
        status = SecTrustEvaluate(myTrust, &trustResult);       // 4
    
    //...                                                             
    if (trustResult == kSecTrustResultConfirm || trustResult == kSecTrustResultProceed || trustResult == kSecTrustResultUnspecified)                           // 5
    
        certificateHandler(YES);
    

    // ...
    if (myPolicy)
    CFRelease(myPolicy);  

【讨论】:

我不明白这一行。 NSString *thePath = [[NSBundle mainBundle] pathForResource:@"Romeo Montague" ofType:@"cer"];我需要在设备中安装 cer 吗?

以上是关于为 MCSession 授权证书的主要内容,如果未能解决你的问题,请参考以下文章

从 MCSession 断开单个对等点?

MCSession 的距离范围是多少?

MCSession 需要很长时间才能解除分配

iOS 8 支持的最大 MCSession 数

MCSession 发送资源取消

MCSession 正在连接,但从未连接