/*
 * 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,
          @"No info is retrieved from Keychain OSStatus: %d with the 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
