使用Swift向iOS钥匙串添加项目和查询

我无法将所有可用于添加数据的Objective C代码示例转换为从iOS钥匙串查询数据到 Swift.我试图做一个字符串(访问令牌)的基本存储并读回来.我已经看了Stack Overflow上的其他一些问题,但我不能完全相信它.我试图从各种来源拼凑一个解决方案.

编辑1:我尝试了一个更基本的设置,因为我以为我的self.defaultKeychainQuery可能已经搞砸了.我已将以下代码更新为最新版本.

编辑2:得到它的工作.我没有正确地将数据值添加到保存查询.我需要将字符串转换为NSData.我已将以下代码更新为最新的工作版本.

编辑3:由于Xerxes在下面指出,因为字典的一些问题,此代码不适用于高于Beta 1的Xcode版本.如果你知道这是一个修复,请让我知道.

更新:我把它变成了keychain library written in Swift called Locksmith.

保存

class func save(service: Nsstring,data: Nsstring) {
  var dataFromString: NSData = data.dataUsingEncoding(NSUTF8StringEncoding,allowLossyConversion: false)
  // Instantiate a new default keychain query
  var keychainQuery: NSMutableDictionary = NSMutableDictionary(objects: [kSecclassGenericPassword,service,userAccount,dataFromString],forKeys: [kSecclass,kSecAttrService,kSecAttrAccount,kSecValueData])

  // Delete any existing items
  SecItemDelete(keychainQuery as CFDictionaryRef)

  // Add the new keychain item
  var status: Osstatus = SecItemAdd(keychainQuery as CFDictionaryRef,nil)

  // Check that it worked ok
  println("Saving status code is: \(status)")
}

加载

class func load(service: Nsstring) -> AnyObject? {
    // Instantiate a new default keychain query
    // Tell the query to return a result
    // Limit our results to one item
    var keychainQuery: NSMutableDictionary = NSMutableDictionary(objects: [kSecclassGenericPassword,kcfBooleanTrue,kSecMatchLimitOne],kSecReturnData,kSecMatchLimit])



    // I'm not too sure what's happening here...
    var dataTypeRef :Unmanaged<AnyObject>?

    // Search for the keychain items
    let status: Osstatus = SecItemcopyMatching(keychainQuery,&dataTypeRef)


    println("Loading status code is: \(status)")

    // I'm not too sure what's happening here...
    let opaque = dataTypeRef?.toOpaque()

    if let op = opaque? {
      let retrievedData = Unmanaged<NSData>.fromOpaque(op).takeUnretainedValue()
      println("Retrieved the following data from the keychain: \(retrievedData)")
      var str = Nsstring(data: retrievedData,encoding: NSUTF8StringEncoding)
      println("The decoded string is \(str)")
    } else {
      println("nothing was retrieved from the keychain.")
    }

    return nil
  }

用法(查看控制器)

KeychainService.savetoken("sometoken")
KeychainService.loadToken()

使用这些方便的方法

class func savetoken(token: Nsstring) {
    self.save("service",data: token)
  }

class func loadToken() {
    var token = self.load("service")
    if let t = token {
      println("The token is: \(t)")
    }
  }

这导致控制台中的输出

Saving status code is: 0
Loading status code is: 0
Retrieved the following data from the keychain: <736f6d65 746f6b65 6e>
The decoded string is sometoken

非常感谢你的帮助.我不太确定一旦获得了它,或者如果它有任何数据给上面的代码,该如何处理dataTypeRef.

为了使其正常工作,您将需要检索钥匙串常量的保留值并存储,首先如下所示:
let kSecclassValue = kSecclass.takeRetainedValue() as Nsstring
let kSecAttrAccountValue = kSecAttrAccount.takeRetainedValue() as Nsstring
let kSecValueDataValue = kSecValueData.takeRetainedValue() as Nsstring
let kSecclassGenericPasswordValue = kSecclassGenericPassword.takeRetainedValue() as Nsstring
let kSecAttrServiceValue = kSecAttrService.takeRetainedValue() as Nsstring
let kSecMatchLimitValue = kSecMatchLimit.takeRetainedValue() as Nsstring
let kSecReturnDataValue = kSecReturnData.takeRetainedValue() as Nsstring
let kSecMatchLimitOneValue = kSecMatchLimitOne.takeRetainedValue() as Nsstring

然后,您可以引用NSMutableDictionary中的值,如下所示:

var keychainQuery: NSMutableDictionary = NSMutableDictionary(objects: [kSecclassGenericPasswordValue,kSecMatchLimitOneValue],forKeys: [kSecclassValue,kSecAttrServiceValue,kSecAttrAccountValue,kSecReturnDataValue,kSecMatchLimitValue])

我写了一篇关于它的博文:
http://rshelby.com/2014/08/using-swift-to-save-and-query-ios-keychain-in-xcode-beta-4/

希望这可以帮助!

rshelby

相关文章

软件简介:蓝湖辅助工具,减少移动端开发中控件属性的复制和粘...
现实生活中,我们听到的声音都是时间连续的,我们称为这种信...
前言最近在B站上看到一个漂亮的仙女姐姐跳舞视频,循环看了亿...
【Android App】实战项目之仿抖音的短视频分享App(附源码和...
前言这一篇博客应该是我花时间最多的一次了,从2022年1月底至...
因为我既对接过session、cookie,也对接过JWT,今年因为工作...