仅使用服务名称获取存储在钥匙串中的用户名?或者:您应该在哪里存储用户名?
因此,OS X 钥匙串包含三部分信息:
- ServiceName(我的应用程序的名称)
- 用户名
- 密码
我显然总是知道 ServiceName。有没有办法找到该 ServiceName 的任何已保存的用户名? (一旦知道用户名,找到密码就很容易了。)
我更喜欢使用漂亮的 Cocoa 包装器,例如 EMKeychain 来执行此操作。但 EMKeychain 需要用户名才能获取任何钥匙串项目!
+ (EMGenericKeychainItem *)genericKeychainItemForService:(NSString *)serviceNameString withUsername:(NSString *)usernameString;
如果您需要用户名来查找凭据,您希望如何充分利用钥匙串中的保存凭据?最佳实践是将用户名保存在 .plist 文件中还是其他文件中?
So the OS X Keychain has three pieces of information:
- ServiceName (the name of my app)
- Username
- Password
I obviously always know the ServiceName. Is there a way to find any saved Username(s) for that ServiceName? (Finding the password is easy once you know the Username.)
I would much prefer to use a nice Cocoa wrapper such as EMKeychain to do this. But EMKeychain requires the UserName to get any keychain item!
+ (EMGenericKeychainItem *)genericKeychainItemForService:(NSString *)serviceNameString withUsername:(NSString *)usernameString;
How are you expected to fully utilize saving credentials in the Keychain, if you need the Username to find the credentials? Is the best practice to save the Username in the .plist file or something?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(3)
SecKeychainFindGenericPassword
仅返回单个钥匙串项。要查找特定服务的所有通用密码,您需要在钥匙串上运行查询。根据您的目标 OS X 版本,有多种方法可以实现此目的。如果您需要在 10.5 或更低版本上运行,则需要使用
SecKeychainSearchCreateFromAttributes
。这是一个相当可怕的 API。这是返回将用户名映射到密码的字典的方法的粗略版本。对于 10.6 及更高版本,您可以使用不太方便的
SecItemCopyMatching
API:对于 10.7 或更高版本,您可以使用我精彩的 LKKeychain 框架(PLUG!)。它不支持构建基于属性的查询,但您可以简单地列出所有密码并过滤掉不需要的密码。
(我没有尝试运行,甚至没有编译任何上述代码示例;对于任何拼写错误,我深表歉意。)
SecKeychainFindGenericPassword
only returns a single keychain item. To find all generic passwords for a specific service, you need to run a query on the keychain. There are several ways to do this, based on what version of OS X you target.If you need to run on 10.5 or below, you'll need to use
SecKeychainSearchCreateFromAttributes
. It's a rather horrible API. Here is a rough cut of a method that returns a dictionary mapping usernames to passwords.For 10.6 and later, you can use the somewhat less inconvenient
SecItemCopyMatching
API:For 10.7 or later, you can use my wonderful LKKeychain framework (PLUG!). It doesn't support building attribute-based queries, but you can simply list all passwords and filter out the ones you don't need.
(I didn't try running, or even compiling any of the above code samples; sorry for any typos.)
您不需要用户名。您可以使用 EMKeychain,但这是该类强加的人为区别; 底层钥匙串服务功能不需要用户名来查找钥匙串项目。
直接使用
SecKeychainFindGenericPassword
时,为用户名参数传递0
和NULL
。它将返回该服务上存在的a钥匙串项。但是,这只会返回一项。如果用户在同一服务上有多个钥匙串项目,您将不知道这一点,也不知道您获得了哪一个(文档说它返回“第一个”匹配项目,但没有说明什么是“第一个”)。如果您想要该服务的任何和所有项目,您应该 创建搜索并使用它。
You don't need the username. You do with EMKeychain, but that's an artificial distinction that that class imposes; the underlying Keychain Services function does not require a username to find a keychain item.
When using
SecKeychainFindGenericPassword
directly, pass0
andNULL
for the username parameters. It will return a keychain item that exists on that service.However, that will return only one item. If the user has multiple keychain items on the same service, you won't know that, or which one you got (the documentation says it returns the “first” matching item, with no specification of what it considers “first”). If you want any and all items for that service, you should create a search and use that.
通用密码具有服务名称和用户名的唯一密钥。因此,要获取单个通用钥匙串条目,您需要提供两者。但是,您可以使用 SecKeychainFindGenericPassword 函数迭代给定服务的所有通用钥匙串条目。
(免责声明:我对在 EMKeychain 中执行此操作一无所知。)
Generic passwords have a unique key of the service name and the username. Thus, to fetch a single generic keychain entry, you will need to provide both. However, you can iterate over all generic keychain entries for your given service using the
SecKeychainFindGenericPassword
function.(Disclaimer: I don't know anything about doing this in EMKeychain.)