175 lines
6.5 KiB
Objective-C
175 lines
6.5 KiB
Objective-C
/*
|
|
* Copyright 2019 Google
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#import "FIRInstanceIDKeychain.h"
|
|
|
|
#import "FIRInstanceIDKeyPair.h"
|
|
#import "FIRInstanceIDKeyPairUtilities.h"
|
|
#import "FIRInstanceIDLogger.h"
|
|
|
|
NSString *const kFIRInstanceIDKeychainErrorDomain = @"com.google.iid";
|
|
|
|
static const NSUInteger kRSA2048KeyPairSize = 2048;
|
|
|
|
@interface FIRInstanceIDKeychain () {
|
|
dispatch_queue_t _keychainOperationQueue;
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation FIRInstanceIDKeychain
|
|
|
|
+ (instancetype)sharedInstance {
|
|
static FIRInstanceIDKeychain *sharedInstance;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
sharedInstance = [[FIRInstanceIDKeychain alloc] init];
|
|
});
|
|
return sharedInstance;
|
|
}
|
|
|
|
- (instancetype)init {
|
|
self = [super init];
|
|
if (self) {
|
|
_keychainOperationQueue =
|
|
dispatch_queue_create("com.google.FirebaseInstanceID.Keychain", DISPATCH_QUEUE_SERIAL);
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (CFTypeRef)itemWithQuery:(NSDictionary *)keychainQuery {
|
|
__block SecKeyRef keyRef = NULL;
|
|
dispatch_sync(_keychainOperationQueue, ^{
|
|
OSStatus status =
|
|
SecItemCopyMatching((__bridge CFDictionaryRef)keychainQuery, (CFTypeRef *)&keyRef);
|
|
|
|
if (status != noErr) {
|
|
if (keyRef) {
|
|
CFRelease(keyRef);
|
|
}
|
|
FIRInstanceIDLoggerDebug(kFIRInstanceIDKeychainReadItemError,
|
|
@"Info is not found in Keychain. OSStatus: %d. Keychain query: %@",
|
|
(int)status, keychainQuery);
|
|
}
|
|
});
|
|
return keyRef;
|
|
}
|
|
|
|
- (void)removeItemWithQuery:(NSDictionary *)keychainQuery
|
|
handler:(void (^)(NSError *error))handler {
|
|
dispatch_async(_keychainOperationQueue, ^{
|
|
OSStatus status = SecItemDelete((__bridge CFDictionaryRef)keychainQuery);
|
|
if (status != noErr) {
|
|
FIRInstanceIDLoggerDebug(
|
|
kFIRInstanceIDKeychainDeleteItemError,
|
|
@"Couldn't delete item from Keychain OSStatus: %d with the keychain query %@",
|
|
(int)status, keychainQuery);
|
|
}
|
|
|
|
if (handler) {
|
|
NSError *error;
|
|
// When item is not found, it should NOT be considered as an error. The operation should
|
|
// continue.
|
|
if (status != noErr && status != errSecItemNotFound) {
|
|
error = [NSError errorWithDomain:kFIRInstanceIDKeychainErrorDomain
|
|
code:status
|
|
userInfo:nil];
|
|
}
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
handler(error);
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
- (void)addItemWithQuery:(NSDictionary *)keychainQuery handler:(void (^)(NSError *))handler {
|
|
dispatch_async(_keychainOperationQueue, ^{
|
|
OSStatus status = SecItemAdd((__bridge CFDictionaryRef)keychainQuery, NULL);
|
|
|
|
if (handler) {
|
|
NSError *error;
|
|
if (status != noErr) {
|
|
FIRInstanceIDLoggerWarning(kFIRInstanceIDKeychainAddItemError,
|
|
@"Couldn't add item to Keychain OSStatus: %d", (int)status);
|
|
error = [NSError errorWithDomain:kFIRInstanceIDKeychainErrorDomain
|
|
code:status
|
|
userInfo:nil];
|
|
}
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
handler(error);
|
|
});
|
|
}
|
|
});
|
|
}
|
|
|
|
- (FIRInstanceIDKeyPair *)generateKeyPairWithPrivateTag:(NSString *)privateTag
|
|
publicTag:(NSString *)publicTag {
|
|
// TODO(chliangGoogle) this is called by appInstanceID, which is an internal API used by other
|
|
// Firebase teams, will see if we can make it async.
|
|
NSData *publicTagData = [publicTag dataUsingEncoding:NSUTF8StringEncoding];
|
|
NSData *privateTagData = [privateTag dataUsingEncoding:NSUTF8StringEncoding];
|
|
|
|
NSDictionary *privateKeyAttr = @{
|
|
(__bridge id)kSecAttrIsPermanent : @YES,
|
|
(__bridge id)kSecAttrApplicationTag : privateTagData,
|
|
(__bridge id)kSecAttrLabel : @"Firebase InstanceID Key Pair Private Key",
|
|
(__bridge id)kSecAttrAccessible : (__bridge id)kSecAttrAccessibleAlwaysThisDeviceOnly,
|
|
};
|
|
|
|
NSDictionary *publicKeyAttr = @{
|
|
(__bridge id)kSecAttrIsPermanent : @YES,
|
|
(__bridge id)kSecAttrApplicationTag : publicTagData,
|
|
(__bridge id)kSecAttrLabel : @"Firebase InstanceID Key Pair Public Key",
|
|
(__bridge id)kSecAttrAccessible : (__bridge id)kSecAttrAccessibleAlwaysThisDeviceOnly,
|
|
};
|
|
|
|
NSDictionary *keyPairAttributes = @{
|
|
(__bridge id)kSecAttrKeyType : (__bridge id)kSecAttrKeyTypeRSA,
|
|
(__bridge id)kSecAttrLabel : @"Firebase InstanceID Key Pair",
|
|
(__bridge id)kSecAttrKeySizeInBits : @(kRSA2048KeyPairSize),
|
|
(__bridge id)kSecPrivateKeyAttrs : privateKeyAttr,
|
|
(__bridge id)kSecPublicKeyAttrs : publicKeyAttr,
|
|
};
|
|
|
|
__block SecKeyRef privateKey = NULL;
|
|
__block SecKeyRef publicKey = NULL;
|
|
dispatch_sync(_keychainOperationQueue, ^{
|
|
// SecKeyGeneratePair does not allow you to set kSetAttrAccessible on the keys. We need the keys
|
|
// to be accessible even when the device is locked (i.e. app is woken up during a push
|
|
// notification, or some background refresh).
|
|
OSStatus status =
|
|
SecKeyGeneratePair((__bridge CFDictionaryRef)keyPairAttributes, &publicKey, &privateKey);
|
|
if (status != noErr || publicKey == NULL || privateKey == NULL) {
|
|
FIRInstanceIDLoggerWarning(kFIRInstanceIDKeychainCreateKeyPairError,
|
|
@"Couldn't create keypair from Keychain OSStatus: %d",
|
|
(int)status);
|
|
}
|
|
});
|
|
// Extract the actual public and private key data from the Keychain
|
|
NSDictionary *publicKeyDataQuery = FIRInstanceIDKeyPairQuery(publicTag, YES, YES);
|
|
NSDictionary *privateKeyDataQuery = FIRInstanceIDKeyPairQuery(privateTag, YES, YES);
|
|
|
|
NSData *publicKeyData = (__bridge NSData *)[self itemWithQuery:publicKeyDataQuery];
|
|
NSData *privateKeyData = (__bridge NSData *)[self itemWithQuery:privateKeyDataQuery];
|
|
|
|
return [[FIRInstanceIDKeyPair alloc] initWithPrivateKey:privateKey
|
|
publicKey:publicKey
|
|
publicKeyData:publicKeyData
|
|
privateKeyData:privateKeyData];
|
|
}
|
|
|
|
@end
|