实战 | 将 Android 生物识别身份验证整合至应用中
本文是 Android 生物识别身份验证系列文章的第二篇,上篇文章主要通过比较传统用户名和密码的认证方式和生物识别身份认证方式的不同,以及介绍生物识别加密的不同加密方式,来向开发者展示为何需要在应用中使用生物识别身份认证技术。
BiometricPrompt.PromptInfo https://developer.android.google.cn/reference/androidx/biometric/BiometricPrompt.PromptInfo
setConfirmationRequired(true) https://developer.android.google.cn/reference/androidx/biometric/BiometricPrompt.PromptInfo.Builder#setConfirmationRequired(boolean)
接入生物识别的设计流程
示例中的代码使用了带有 CryptoObject 实例的加密版 BiometricPrompt。
如果您的应用需要认证,那么您就应该创建一个专门的 LoginActivity 组件作为应用的登录界面。无论应用要求进行身份验证的频率多高,只要需要验证,就应该这么做。若用户之前已认证过,那么 LoginActivity 将调用 finish() 方法,让用户继续使用。如果用户还没有进行身份验证,那么您应该检查生物识别身份验证是否启用。
有很多方法来检查是否启用了生物识别。与其在各种不同的替代方案中周旋,不如我们直接深入研究一个特别的方法: 直接检查自定义属性 ciphertextWrapper 是否是 null。当用户在您的应用中启用生物识别身份验证后,您就可以创建一个 CiphertextWrapper 数据类,来将加密后的 userToken (也就是 ciphertext) 存储在 SharedPreferences 或 Room 这样的持久性存储中。因此,若 ciphertextWrapper 不是 null,就相当于您拥有了访问远程服务所需的已加密的 userToken,这也意味着当前生物识别已启用。
if (ciphertextWrapper != null) {
// 用户已启用了生物识别
} else {
// 生物识别未启用
}
binding.useBiometrics.setOnClickListener {
showBiometricPromptForEncryption()
}
....
private fun showBiometricPromptForEncryption() {
val canAuthenticate = BiometricManager.from(applicationContext).canAuthenticate()
if (canAuthenticate == BiometricManager.BIOMETRIC_SUCCESS) {
val secretKeyName = SECRET_KEY_NAME
cryptographyManager = CryptographyManager()
val cipher = cryptographyManager.getInitializedCipherForEncryption(secretKeyName)
val biometricPrompt =
BiometricPromptUtils.createBiometricPrompt(this, ::encryptAndStoreServerToken)
val promptInfo = BiometricPromptUtils.createPromptInfo(this)
biometricPrompt.authenticate(promptInfo, BiometricPrompt.CryptoObject(cipher))
}
}
override fun onResume() {
super.onResume()
if (ciphertextWrapper != null) {
if (SampleAppUser.fakeToken == null) {
showBiometricPromptForDecryption()
} else {
// 用户已经成功登录,因此直接进入接下来的应用流程
// 之后的就交给开发者您来完成了
updateApp(getString(R.string.already_signedin))
}
}
}
....
private fun showBiometricPromptForDecryption() {
ciphertextWrapper?.let { textWrapper ->
val canAuthenticate = BiometricManager.from(applicationContext).canAuthenticate()
if (canAuthenticate == BiometricManager.BIOMETRIC_SUCCESS) {
val secretKeyName = getString(R.string.secret_key_name)
val cipher = cryptographyManager.getInitializedCipherForDecryption(
secretKeyName, textWrapper.initializationVector
)
biometricPrompt =
BiometricPromptUtils.createBiometricPrompt(
this,
::decryptServerTokenFromStorage
)
val promptInfo = BiometricPromptUtils.createPromptInfo(this)
biometricPrompt.authenticate(promptInfo, BiometricPrompt.CryptoObject(cipher))
}
}
}
private fun decryptServerTokenFromStorage(authResult: BiometricPrompt.AuthenticationResult) {
ciphertextWrapper?.let { textWrapper ->
authResult.cryptoObject?.cipher?.let {
val plaintext =
cryptographyManager.decryptData(textWrapper.ciphertext, it)
SampleAppUser.fakeToken = plaintext
// 现在您有了 token,就可以查询服务器上的其他数据了
// 我们之所以称这个为 fakeToken,是因为它并不是真正从服务器中获取到的
// 在真实场景下,您会在从服务器上获取到 token 数据
// 此时,它才能算是一个真正的 token
updateApp(getString(R.string.already_signedin))
}
}
}
完整的蓝图
SharedPreferences
https://developer.android.google.cn/reference/android/content/SharedPreferences
Room
https://developer.android.google.cn/jetpack/androidx/releases/room
Github 上的示例代码 https://github.com/android/security-samples/tree/master/BiometricLoginKotlin
△ 图 5: 使用生物识别同服务器获取授权的完整蓝图
总结
如何扩展 UI 来支持生物识别身份验证; 针对生物识别身份验证的流程,您的应用应着重解决的关键点是什么; 如何设计您的代码来处理生物识别认证的不同场景; 登录流程的完整工程设计图。
推荐阅读