如何实现自动填充凭据提供程序扩展 - iOS 11、swift4



我一直在 iOS 自动填充凭据扩展中工作很长时间。我检查了很多iOS文章和视频。但是我无法在快速类型栏上显示凭据,重置内容已成功集成。谁能给我快速的帮助?

使用从苹果分享的这个视频和网址: https://developer.apple.com/videos/play/wwdc2018/721 https://developer.apple.com/documentation/authenticationservices
我正在使用以下代码将凭据保存到特定域的钥匙串。

let keychain = Keychain(server: "instagram.com", protocolType: .https, authenticationType: .htmlForm)
keychain["emailAddress"] = "Password"

并将此代码用于保存域:

func savedomain(domain: String, account: String, password: String, completion: ((Bool, SharedWebCredentialsManagerError?) -> Void)? = nil) {
SecAddSharedWebCredential(domain as CFString, account as CFString, password as CFString?) { error in
guard let error = error else {
completion?(true, nil)
return
}
let errorDescription = CFErrorCopyDescription(error) as String
let saveFailedError = SharedWebCredentialsManagerError.saveFailed(errorDescription)
completion?(false, saveFailedError)
}
}

我已经创建了自动填充扩展并获取保存的凭据,但无法在 safari 的快速类型栏上显示凭据 instagram.com

我已经为所有社交网站实现了自动填充扩展,共享我的源代码以保存带有域的电子邮件ID密码。

class func save(key: String, data: Data) -> OSStatus {
let query = [
kSecClass as String       : kSecClassGenericPassword as String,
kSecAttrAccount as String : key,
kSecValueData as String   : data ] as [String : Any]
SecItemDelete(query as CFDictionary)
return SecItemAdd(query as CFDictionary, nil)
}

在"保存"函数上面,我已经在我的自定义KeyChainManager类中编写了,我还在KeyChainManager类中添加了以下代码,如下所示。

extension Data {
init<T>(from value: T) {
var value = value
self.init(buffer: UnsafeBufferPointer(start: &value, count: 1))
}
func to<T>(type: T.Type) -> T {
return self.withUnsafeBytes { $0.load(as: T.self) }
}

}

我通过调用我们的KeyChainManager类从VC保存我的数据,如下所示:

let email = (txtEmail?.text ?? "").trimmingCharacters(in: .whitespacesAndNewlines)
let password = (txtPassword?.text ?? "").trimmingCharacters(in: .whitespacesAndNewlines)
let domain = (txtDomain?.text ?? "").lowercased().trimmingCharacters(in: .whitespacesAndNewlines)

let user = User(email: email, password: password, key: self.key, domain: domain, identifier: self.keyIdentifier)
let data = (try? JSONEncoder().encode(user)) ?? Data()
let _ = KeyChain.save(key: "(self.keyIdentifier)", data: data)

所有这些东西都是为了保存我们的凭据,现在重点是如何在我们扩展的CredentialProviderViewController.swift中列出所有保存的凭据。

为此,我在KeyChainManager类中添加了以下方法:

class func load(key: String) -> Data? {

let query = [
kSecClass as String       : kSecClassGenericPassword,
kSecAttrAccount as String : key,
kSecReturnData as String  : kCFBooleanTrue!,
kSecMatchLimit as String  : kSecMatchLimitOne ] as [String : Any]
var dataTypeRef: AnyObject? = nil
let status: OSStatus = SecItemCopyMatching(query as CFDictionary, &dataTypeRef)
if status == noErr {
return dataTypeRef as! Data?
} else {
return nil
}
}

并从扩展的CredentialProviderViewController调用此函数.swift如下所示:

users.removeAll()
for i in 0..<1000000 {
if let encodedData = KeyChain.load(key: "(i)")  {
let user = try! JSONDecoder().decode(User.self, from: encodedData)
if user.key == key && ((serviceIdentifier.contains(user.domain ?? "")) ) {
users.append(user)
}
}else{
break
}
}

我希望此内容对您有所帮助,因为我花了很多天只创建了一个演示:) :) 在下面评论,如果它可以帮助您:) :)

谢谢 安贾利。

您需要填充 ASCredentialIdentityStore 才能使快速类型栏正常工作。请参阅 ASCredentialProviderViewController 的说明:

(可选)将 ASPasswordCredentialIdentity 实例添加到共享的 ASCredentialIdentityStore 中,以使标识直接在 QuickType 栏中可用。

这在您引用的 WWDC 演示文稿中也有描述。

我如何实现 ASCredentialIdentityStore 的人口:

var firstItem = ASPasswordCredentialIdentity(serviceIdentifier: ASCredentialServiceIdentifier(identifier: "https://online.somebank.com/auth/login", type: .URL), user: "login@bank.com", recordIdentifier: nil)
ASCredentialIdentityStore.shared.replaceCredentialIdentities(with: [firstItem]) { result, error in
if result {
print("saved")
}
}

就我而言,一切都很完美。 因此,我从远程获取所有密码,然后用可用密码填充ASCredentialIdentityStore。

相关内容

最新更新