MYKeychain-iPhone.m
author snej@snej.local
Thu Apr 09 21:36:21 2009 -0700 (2009-04-09)
changeset 4 f4709533c816
parent 2 8982b8fada63
child 5 b2e360b78189
permissions -rw-r--r--
* Certificate signing!!!
* MYIdentity class.
     1 //
     2 //  MYKeychain-iPhone.m
     3 //  MYCrypto-iPhone
     4 //
     5 //  Created by Jens Alfke on 3/31/09.
     6 //  Copyright 2009 Jens Alfke. All rights reserved.
     7 //
     8 
     9 #import "MYCrypto_Private.h"
    10 #import "MYDigest.h"
    11 
    12 #if MYCRYPTO_USE_IPHONE_API
    13 
    14 
    15 @interface MYKeyEnumerator : NSEnumerator
    16 {
    17     CFArrayRef _results;
    18     CFTypeRef _itemClass;
    19     CFIndex _index;
    20 }
    21 
    22 - (id) initWithQuery: (NSMutableDictionary*)query;
    23 + (id) firstItemWithQuery: (NSMutableDictionary*)query;
    24 @end
    25 
    26 
    27 
    28 @implementation MYKeychain
    29 
    30 
    31 + (MYKeychain*) allKeychains
    32 {
    33     // iPhone only has a single keychain.
    34     return [self defaultKeychain];
    35 }
    36 
    37 + (MYKeychain*) defaultKeychain
    38 {
    39     static MYKeychain *sDefaultKeychain;
    40     @synchronized(self) {
    41         if (!sDefaultKeychain) {
    42             sDefaultKeychain = [[self alloc] init];
    43         }
    44     }
    45     return sDefaultKeychain;
    46 }
    47 
    48 
    49 - (id) copyWithZone: (NSZone*)zone {
    50     // It's not necessary to make copies of Keychain objects. This makes it more efficient
    51     // to use instances as NSDictionary keys or store them in NSSets.
    52     return [self retain];
    53 }
    54 
    55 
    56 
    57 #pragma mark -
    58 #pragma mark SEARCHING:
    59 
    60 
    61 - (MYPublicKey*) publicKeyWithDigest: (MYSHA1Digest*)pubKeyDigest {
    62     return [MYKeyEnumerator firstItemWithQuery:
    63                 $mdict({(id)kSecClass, (id)kSecClassKey},
    64                       {(id)kSecAttrPublicKeyHash, pubKeyDigest.asData},
    65                       {(id)kSecReturnRef, $true})];
    66 }   
    67 
    68 - (NSEnumerator*) enumeratePublicKeys {
    69     NSMutableDictionary *query = $mdict({(id)kSecClass, (id)kSecClassKey},
    70                                 {(id)kSecAttrKeyClass, (id)kSecAttrKeyClassPublic},
    71                                 {(id)kSecMatchLimit, (id)kSecMatchLimitAll},
    72                                 {(id)kSecReturnRef, $true});
    73     return [[[MYKeyEnumerator alloc] initWithQuery: query] autorelease];
    74 }
    75 
    76 
    77 - (MYPrivateKey*) privateKeyWithDigest: (MYSHA1Digest*)pubKeyDigest {
    78     return [MYKeyEnumerator firstItemWithQuery:
    79                 $mdict({(id)kSecClass, (id)kSecClassKey},
    80                       {(id)kSecAttrKeyClass, (id)kSecAttrKeyClassPrivate},
    81                       {(id)kSecAttrPublicKeyHash, pubKeyDigest.asData},
    82                       {(id)kSecReturnRef, $true})];
    83 }
    84 
    85 - (NSEnumerator*) enumeratePrivateKeys {
    86     NSMutableDictionary *query = $mdict({(id)kSecClass, (id)kSecClassKey},
    87                                 {(id)kSecAttrKeyClass, (id)kSecAttrKeyClassPrivate},
    88                                 {(id)kSecMatchLimit, (id)kSecMatchLimitAll},
    89                                 {(id)kSecReturnRef, $true});
    90     return [[[MYKeyEnumerator alloc] initWithQuery: query] autorelease];
    91 }
    92 
    93 - (MYCertificate*) certificateWithDigest: (MYSHA1Digest*)pubKeyDigest {
    94     return [MYKeyEnumerator firstItemWithQuery:
    95                 $mdict({(id)kSecClass, (id)kSecClassCertificate},
    96                       {(id)kSecAttrPublicKeyHash, pubKeyDigest.asData},
    97                       {(id)kSecReturnRef, $true})];
    98 }
    99 
   100 - (NSEnumerator*) enumerateCertificates {
   101     NSMutableDictionary *query = $mdict({(id)kSecClass, (id)kSecClassCertificate},
   102                                 {(id)kSecMatchLimit, (id)kSecMatchLimitAll},
   103                                 {(id)kSecReturnRef, $true});
   104     return [[[MYKeyEnumerator alloc] initWithQuery: query] autorelease];
   105 }
   106 
   107 - (NSEnumerator*) enumerateSymmetricKeys {
   108     NSMutableDictionary *query = $mdict({(id)kSecClass, (id)kSecClassKey},
   109                                 {(id)kSecAttrKeyClass, (id)kSecAttrKeyClassSymmetric},
   110                                 {(id)kSecMatchLimit, (id)kSecMatchLimitAll},
   111                                 {(id)kSecReturnRef, $true});
   112     return [[[MYKeyEnumerator alloc] initWithQuery: query] autorelease];
   113 }
   114 
   115 - (NSEnumerator*) symmetricKeysWithAlias: (NSString*)alias {
   116     NSMutableDictionary *query = $mdict({(id)kSecClass, (id)kSecClassKey},
   117                                 {(id)kSecAttrKeyClass, (id)kSecAttrKeyClassSymmetric},
   118                                 {(id)kSecAttrApplicationTag, alias},
   119                                 {(id)kSecMatchLimit, (id)kSecMatchLimitAll},
   120                                 {(id)kSecReturnRef, $true});
   121     return [[[MYKeyEnumerator alloc] initWithQuery: query] autorelease];
   122 }
   123 
   124 
   125 #pragma mark -
   126 #pragma mark IMPORT:
   127 
   128 
   129 - (MYPublicKey*) importPublicKey: (NSData*)keyData {
   130     return [[[MYPublicKey alloc] _initWithKeyData: keyData 
   131                                       forKeychain: self]
   132             autorelease];
   133 }
   134 
   135 - (MYCertificate*) importCertificate: (NSData*)data
   136 {
   137     Assert(data);
   138     NSDictionary *info = $dict( {(id)kSecClass, (id)kSecClassCertificate},
   139                                 {(id)kSecValueData, data},
   140                                 {(id)kSecReturnRef, $true} );
   141     SecCertificateRef cert;
   142     if (!check(SecItemAdd((CFDictionaryRef)info, (CFTypeRef*)&cert), @"SecItemAdd"))
   143         return nil;
   144     return [[[MYCertificate alloc] initWithCertificateRef: cert] autorelease];
   145 }
   146 
   147 
   148 #pragma mark -
   149 #pragma mark GENERATION:
   150 
   151 
   152 - (MYSymmetricKey*) generateSymmetricKeyOfSize: (unsigned)keySizeInBits
   153                                      algorithm: (CCAlgorithm)algorithm
   154 {
   155     return [MYSymmetricKey _generateSymmetricKeyOfSize: keySizeInBits
   156                                              algorithm: algorithm inKeychain: self];
   157 }
   158 
   159 - (MYPrivateKey*) generateRSAKeyPairOfSize: (unsigned)keySize {
   160     return [MYPrivateKey _generateRSAKeyPairOfSize: keySize inKeychain: self];
   161 }
   162 
   163 
   164 
   165 @end
   166 
   167 
   168 
   169 #pragma mark -
   170 @implementation MYKeyEnumerator
   171 
   172 - (id) initWithQuery: (NSMutableDictionary*)query {
   173     self = [super init];
   174     if (self) {
   175         if (![query objectForKey: (id)kSecMatchLimit])
   176             [query setObject: (id)kSecMatchLimitAll forKey: (id)kSecMatchLimit];
   177         OSStatus err = SecItemCopyMatching((CFDictionaryRef)query, (CFTypeRef*)&_results);
   178         if (err && err != errSecItemNotFound) {
   179             check(err,@"SecItemCopyMatching");
   180             [self release];
   181             return nil;
   182         }
   183         if (_results) CFRetain(_results);
   184         _itemClass = (CFTypeRef)[query objectForKey: (id)kSecClass];
   185         if (_itemClass == kSecClassKey)
   186             _itemClass = (CFTypeRef)[query objectForKey: (id)kSecAttrKeyClass];
   187         if (_itemClass) CFRetain(_itemClass);
   188     }
   189     return self;
   190 }
   191 
   192 + (id) firstItemWithQuery: (NSMutableDictionary*)query {
   193     MYKeyEnumerator *e = [[self alloc] initWithQuery: query];
   194     MYKeychainItem *item = e.nextObject;
   195     [e release];
   196     return item;
   197 }    
   198 
   199 - (void) dealloc
   200 {
   201     if (_itemClass) CFRelease(_itemClass);
   202     if (_results) CFRelease(_results);
   203     [super dealloc];
   204 }
   205 
   206 
   207 - (id) nextObject {
   208     if (!_results)
   209         return nil;
   210     MYKeychainItem *next = nil;
   211     while (next==nil && _index < CFArrayGetCount(_results)) {
   212         CFTypeRef found = CFArrayGetValueAtIndex(_results, _index++); 
   213         if (_itemClass == kSecAttrKeyClassPrivate) {
   214             next = [[MYPrivateKey alloc] initWithKeyRef: (SecKeyRef)found];
   215         } else if (_itemClass == kSecAttrKeyClassPublic) {
   216             next = [[[MYPublicKey alloc] initWithKeyRef: (SecKeyRef)found] autorelease];
   217         } else if (_itemClass == kSecAttrKeyClassSymmetric) {
   218             next = [[[MYSymmetricKey alloc] initWithKeyRef: (SecKeyRef)found] autorelease];
   219         } else if (_itemClass == kSecClassCertificate) {
   220             next = [[[MYCertificate alloc] initWithCertificateRef: (SecCertificateRef)found] autorelease];
   221         }
   222         CFRelease(found);
   223     }
   224     return next;
   225 }
   226 
   227 
   228 @end
   229 
   230 #endif MYCRYPTO_USE_IPHONE_API
   231 
   232 
   233 /*
   234  Copyright (c) 2009, Jens Alfke <jens@mooseyard.com>. All rights reserved.
   235  
   236  Redistribution and use in source and binary forms, with or without modification, are permitted
   237  provided that the following conditions are met:
   238  
   239  * Redistributions of source code must retain the above copyright notice, this list of conditions
   240  and the following disclaimer.
   241  * Redistributions in binary form must reproduce the above copyright notice, this list of conditions
   242  and the following disclaimer in the documentation and/or other materials provided with the
   243  distribution.
   244  
   245  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
   246  IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND 
   247  FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRI-
   248  BUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
   249  (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
   250   PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 
   251  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 
   252  THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
   253  */