MYKeychain-iPhone.m
author snej@snej.local
Sat Apr 04 22:56:13 2009 -0700 (2009-04-04)
changeset 1 60e4cbbb5128
child 2 8982b8fada63
permissions -rw-r--r--
Code cleanup, more header comments.
     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 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 - (MYKeyPair*) keyPairWithDigest: (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*) enumerateKeyPairs {
    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 - (MYKeyPair*) generateRSAKeyPairOfSize: (unsigned)keySize {
   160     return [MYKeyPair _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     for (; next==nil && _index < CFArrayGetCount(_results); _index++) {
   212         CFTypeRef found = CFArrayGetValueAtIndex(_results, _index); 
   213         if (_itemClass == kSecAttrKeyClassPrivate) {
   214             MYSHA1Digest *digest = [MYPublicKey _digestOfKey: (SecKeyRef)found];
   215             if (digest) {
   216                 MYPublicKey *publicKey = [[MYKeychain defaultKeychain] publicKeyWithDigest: digest];
   217                 if (publicKey)
   218                     next = [[[MYKeyPair alloc] initWithPublicKeyRef: publicKey.keyRef
   219                                                      privateKeyRef: (SecKeyRef)found]
   220                            autorelease];
   221                 else {
   222                     // The matching public key won't turn up if it's embedded in a certificate;
   223                     // I'd have to search for certs if I wanted to look that up. Skip it for now.
   224                     //Warn(@"Couldn't find matching public key for private key!");
   225                 }
   226             }
   227             break;
   228         } else if (_itemClass == kSecAttrKeyClassPublic) {
   229             next = [[[MYPublicKey alloc] initWithKeyRef: (SecKeyRef)found] autorelease];
   230             break;
   231         } else if (_itemClass == kSecAttrKeyClassSymmetric) {
   232             next = [[[MYSymmetricKey alloc] initWithKeyRef: (SecKeyRef)found] autorelease];
   233             break;
   234         } else if (_itemClass == kSecClassCertificate) {
   235             next = [[[MYCertificate alloc] initWithCertificateRef: (SecCertificateRef)found] autorelease];
   236             break;
   237         }
   238         CFRelease(found);
   239     }
   240     return next;
   241 }
   242 
   243 
   244 @end
   245 
   246 #endif USE_IPHONE_API
   247 
   248 
   249 /*
   250  Copyright (c) 2009, Jens Alfke <jens@mooseyard.com>. All rights reserved.
   251  
   252  Redistribution and use in source and binary forms, with or without modification, are permitted
   253  provided that the following conditions are met:
   254  
   255  * Redistributions of source code must retain the above copyright notice, this list of conditions
   256  and the following disclaimer.
   257  * Redistributions in binary form must reproduce the above copyright notice, this list of conditions
   258  and the following disclaimer in the documentation and/or other materials provided with the
   259  distribution.
   260  
   261  THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
   262  IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND 
   263  FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRI-
   264  BUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
   265  (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
   266   PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 
   267  CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 
   268  THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
   269  */