MYKey.m
author snej@snej.local
Sat Apr 04 20:42:03 2009 -0700 (2009-04-04)
changeset 0 0a6527af039b
child 1 60e4cbbb5128
permissions -rw-r--r--
Initial checkin. Passes tests on Mac and in iPhone simulator.
snej@0
     1
//
snej@0
     2
//  MYKey.m
snej@0
     3
//  MYCrypto
snej@0
     4
//
snej@0
     5
//  Created by Jens Alfke on 3/21/09.
snej@0
     6
//  Copyright 2009 Jens Alfke. All rights reserved.
snej@0
     7
//
snej@0
     8
snej@0
     9
#import "MYKey.h"
snej@0
    10
#import "MYCrypto_Private.h"
snej@0
    11
#import "MYDigest.h"
snej@0
    12
#import "MYErrorUtils.h"
snej@0
    13
snej@0
    14
#if !USE_IPHONE_API
snej@0
    15
snej@0
    16
snej@0
    17
#pragma mark -
snej@0
    18
@implementation MYKey
snej@0
    19
snej@0
    20
snej@0
    21
- (id) initWithKeyRef: (SecKeyRef)key {
snej@0
    22
    self = [super initWithKeychainItemRef: (SecKeychainItemRef)key];
snej@0
    23
    if (self) {
snej@0
    24
        _key = key;     // superclass has already CFRetained it
snej@0
    25
    }
snej@0
    26
    return self;
snej@0
    27
}
snej@0
    28
snej@0
    29
- (id) _initWithKeyData: (NSData*)data
snej@0
    30
            forKeychain: (SecKeychainRef)keychain {
snej@0
    31
    SecKeyImportExportParameters params = {};
snej@0
    32
    SecKeyRef key = importKey(data, self.keyType, keychain, &params);
snej@0
    33
    if (!key) {
snej@0
    34
        [self release];
snej@0
    35
        return nil;
snej@0
    36
    }
snej@0
    37
    self = [super initWithKeychainItemRef: (SecKeychainItemRef)key];
snej@0
    38
    if (self) {
snej@0
    39
        _key = key;
snej@0
    40
    }
snej@0
    41
    CFRelease(key);
snej@0
    42
    return self;
snej@0
    43
}
snej@0
    44
snej@0
    45
- (id) initWithKeyData: (NSData*)data {
snej@0
    46
    return [self _initWithKeyData: data forKeychain: nil];
snej@0
    47
}
snej@0
    48
snej@0
    49
snej@0
    50
- (NSString*) description {
snej@0
    51
    return $sprintf(@"%@[%p]", [self class], _key);     //FIX: Can we do anything better?
snej@0
    52
}
snej@0
    53
snej@0
    54
snej@0
    55
- (SecExternalItemType) keyType {
snej@0
    56
    AssertAbstractMethod();
snej@0
    57
}
snej@0
    58
snej@0
    59
snej@0
    60
@synthesize keyRef=_key;
snej@0
    61
snej@0
    62
snej@0
    63
- (MYKey*) asKey {
snej@0
    64
    return self;
snej@0
    65
}
snej@0
    66
snej@0
    67
- (const CSSM_KEY*) cssmKey {
snej@0
    68
    const CSSM_KEY *cssmKey = NULL;
snej@0
    69
    Assert(check(SecKeyGetCSSMKey(_key, &cssmKey), @"SecKeyGetCSSMKey"), @"Failed to get CSSM_KEY");
snej@0
    70
    return cssmKey;
snej@0
    71
}
snej@0
    72
snej@0
    73
- (NSData*) exportKeyInFormat: (SecExternalFormat)format withPEM: (BOOL)withPEM {
snej@0
    74
    CFDataRef data = NULL;
snej@0
    75
    if (check(SecKeychainItemExport(_key, format, (withPEM ?kSecItemPemArmour :0), NULL, &data),
snej@0
    76
              @"SecKeychainItemExport"))
snej@0
    77
        return [(id)CFMakeCollectable(data) autorelease];
snej@0
    78
    else
snej@0
    79
        return nil;
snej@0
    80
}
snej@0
    81
snej@0
    82
- (NSData*) keyData {
snej@0
    83
    return [self exportKeyInFormat: kSecFormatRawKey withPEM: NO];
snej@0
    84
}
snej@0
    85
snej@0
    86
- (NSString*) name {
snej@0
    87
    return [self stringValueOfAttribute: kSecKeyPrintName];
snej@0
    88
}
snej@0
    89
snej@0
    90
- (void) setName: (NSString*)name {
snej@0
    91
    [self setValue: name ofAttribute: kSecKeyPrintName];
snej@0
    92
}
snej@0
    93
snej@0
    94
- (NSString*) comment {
snej@0
    95
    return [self stringValueOfAttribute: kSecKeyApplicationTag];
snej@0
    96
}
snej@0
    97
snej@0
    98
- (void) setComment: (NSString*)comment {
snej@0
    99
    [self setValue: comment ofAttribute: kSecKeyApplicationTag];
snej@0
   100
}
snej@0
   101
snej@0
   102
- (NSString*) alias {
snej@0
   103
    return [self stringValueOfAttribute: kSecKeyAlias];
snej@0
   104
}
snej@0
   105
snej@0
   106
- (void) setAlias: (NSString*)alias {
snej@0
   107
    [self setValue: alias ofAttribute: kSecKeyAlias];
snej@0
   108
}
snej@0
   109
snej@0
   110
snej@0
   111
@end
snej@0
   112
snej@0
   113
snej@0
   114
snej@0
   115
snej@0
   116
#pragma mark -
snej@0
   117
#pragma mark UTILITY FUNCTIONS:
snej@0
   118
snej@0
   119
snej@0
   120
SecKeyRef importKey(NSData *data, 
snej@0
   121
                    SecExternalItemType type,
snej@0
   122
                    SecKeychainRef keychain,
snej@0
   123
                    SecKeyImportExportParameters *params) {
snej@0
   124
    SecExternalFormat inputFormat = (type==kSecItemTypeSessionKey) ?kSecFormatRawKey :kSecFormatOpenSSL;
snej@0
   125
    CFArrayRef items = NULL;
snej@0
   126
    
snej@0
   127
    params->version = SEC_KEY_IMPORT_EXPORT_PARAMS_VERSION;
snej@0
   128
    params->flags |= kSecKeyImportOnlyOne;
snej@0
   129
    if (keychain) {
snej@0
   130
        params->keyAttributes = CSSM_KEYATTR_EXTRACTABLE | CSSM_KEYATTR_PERMANENT;
snej@0
   131
        if (type==kSecItemTypeSessionKey)
snej@0
   132
            params->keyUsage = CSSM_KEYUSE_ENCRYPT | CSSM_KEYUSE_DECRYPT;
snej@0
   133
        else if (type==kSecItemTypePublicKey)
snej@0
   134
            params->keyUsage = CSSM_KEYUSE_ENCRYPT | CSSM_KEYUSE_VERIFY;
snej@0
   135
        else if (type==kSecItemTypePrivateKey)
snej@0
   136
            params->keyUsage = CSSM_KEYUSE_DECRYPT | CSSM_KEYUSE_SIGN;
snej@0
   137
    }
snej@0
   138
    if (!check(SecKeychainItemImport((CFDataRef)data, NULL, &inputFormat, &type,
snej@0
   139
                                     0, params, keychain, &items),
snej@0
   140
               @"SecKeychainItemImport"))
snej@0
   141
        return nil;
snej@0
   142
    if (!items || CFArrayGetCount(items) != 1)
snej@0
   143
        return nil;
snej@0
   144
    SecKeyRef key = (SecKeyRef)CFRetain(CFArrayGetValueAtIndex(items,0));
snej@0
   145
    CFRelease(items);
snej@0
   146
    return key; // caller must CFRelease
snej@0
   147
}    
snej@0
   148
snej@0
   149
snej@0
   150
#endif USE_IPHONE_API
snej@0
   151
snej@0
   152
snej@0
   153
snej@0
   154
/*
snej@0
   155
 Copyright (c) 2009, Jens Alfke <jens@mooseyard.com>. All rights reserved.
snej@0
   156
 
snej@0
   157
 Redistribution and use in source and binary forms, with or without modification, are permitted
snej@0
   158
 provided that the following conditions are met:
snej@0
   159
 
snej@0
   160
 * Redistributions of source code must retain the above copyright notice, this list of conditions
snej@0
   161
 and the following disclaimer.
snej@0
   162
 * Redistributions in binary form must reproduce the above copyright notice, this list of conditions
snej@0
   163
 and the following disclaimer in the documentation and/or other materials provided with the
snej@0
   164
 distribution.
snej@0
   165
 
snej@0
   166
 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
snej@0
   167
 IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND 
snej@0
   168
 FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRI-
snej@0
   169
 BUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
snej@0
   170
 (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
snej@0
   171
  PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 
snej@0
   172
 CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 
snej@0
   173
 THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
snej@0
   174
 */