From 6e9ccc593f40b0908fa2293eee5caabfe89f2c06 Mon Sep 17 00:00:00 2001 From: Dominik Dzienia Date: Tue, 24 Mar 2020 22:22:23 +0100 Subject: [PATCH 1/6] Support for master password and storing password as hashes (HMAC) instead of plaintext, additional crypto utils with tests (partialy for in-progress pref export encryption), changed PasswordCheck to more common UI and to use lambdas (will need given password in lambda callback for prefs enc) --- .../activities/MyPreferenceFragment.kt | 45 +++++- .../nightscout/androidaps/utils/CryptoUtil.kt | 142 ++++++++++++++++++ .../utils/protection/PasswordCheck.kt | 60 +++++++- .../utils/protection/ProtectionCheck.kt | 12 +- app/src/main/res/drawable/ic_key.xml | 9 ++ app/src/main/res/drawable/ic_key_48dp.xml | 17 +++ app/src/main/res/layout/passwordprompt.xml | 10 +- app/src/main/res/values/protection.xml | 13 +- app/src/main/res/xml/pref_general.xml | 12 +- .../androidaps/utils/CryptoUtilTest.kt | 76 ++++++++++ 10 files changed, 369 insertions(+), 27 deletions(-) create mode 100644 app/src/main/java/info/nightscout/androidaps/utils/CryptoUtil.kt create mode 100644 app/src/main/res/drawable/ic_key.xml create mode 100644 app/src/main/res/drawable/ic_key_48dp.xml create mode 100644 app/src/test/java/info/nightscout/androidaps/utils/CryptoUtilTest.kt diff --git a/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt b/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt index 86bee1bca2..122ec8d5f0 100644 --- a/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt +++ b/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt @@ -50,8 +50,10 @@ import info.nightscout.androidaps.plugins.source.EversensePlugin import info.nightscout.androidaps.plugins.source.GlimpPlugin import info.nightscout.androidaps.plugins.source.PoctechPlugin import info.nightscout.androidaps.plugins.source.TomatoPlugin +import info.nightscout.androidaps.utils.CryptoUtil import info.nightscout.androidaps.utils.OKDialog.show import info.nightscout.androidaps.utils.SafeParse +import info.nightscout.androidaps.utils.protection.PasswordCheck import info.nightscout.androidaps.utils.protection.ProtectionCheck import info.nightscout.androidaps.utils.resources.ResourceHelper import info.nightscout.androidaps.utils.sharedPreferences.SP @@ -98,6 +100,8 @@ class MyPreferenceFragment : PreferenceFragmentCompat(), OnSharedPreferenceChang @Inject lateinit var wearPlugin: WearPlugin @Inject lateinit var maintenancePlugin: MaintenancePlugin + @Inject lateinit var passwordCheck: PasswordCheck + @Inject lateinit var androidInjector: DispatchingAndroidInjector override fun androidInjector(): AndroidInjector = androidInjector @@ -254,19 +258,19 @@ class MyPreferenceFragment : PreferenceFragmentCompat(), OnSharedPreferenceChang // Preferences if (pref.getKey() == resourceHelper.gs(R.string.key_settings_protection)) { val pass: Preference? = findPreference(resourceHelper.gs(R.string.key_settings_password)) - if (pass != null) pass.isEnabled = pref.value == ProtectionCheck.ProtectionType.PASSWORD.ordinal.toString() + if (pass != null) pass.isEnabled = pref.value == ProtectionCheck.ProtectionType.CUSTOM_PASSWORD.ordinal.toString() } // Application // Application if (pref.getKey() == resourceHelper.gs(R.string.key_application_protection)) { val pass: Preference? = findPreference(resourceHelper.gs(R.string.key_application_password)) - if (pass != null) pass.isEnabled = pref.value == ProtectionCheck.ProtectionType.PASSWORD.ordinal.toString() + if (pass != null) pass.isEnabled = pref.value == ProtectionCheck.ProtectionType.CUSTOM_PASSWORD.ordinal.toString() } // Bolus // Bolus if (pref.getKey() == resourceHelper.gs(R.string.key_bolus_protection)) { val pass: Preference? = findPreference(resourceHelper.gs(R.string.key_bolus_password)) - if (pass != null) pass.isEnabled = pref.value == ProtectionCheck.ProtectionType.PASSWORD.ordinal.toString() + if (pass != null) pass.isEnabled = pref.value == ProtectionCheck.ProtectionType.CUSTOM_PASSWORD.ordinal.toString() } } if (pref is EditTextPreference) { @@ -281,6 +285,16 @@ class MyPreferenceFragment : PreferenceFragmentCompat(), OnSharedPreferenceChang } } } + + if (pref is Preference) { + if ((pref.getKey() != null) && (pref.getKey().contains("_password"))) { + if (sp.getString(pref.getKey(), "").startsWith("hmac:")) { + pref.setSummary("******") + } else { + pref.setSummary(resourceHelper.gs(R.string.password_not_set)) + } + } + } pref?.let { adjustUnitDependentPrefs(it) } } @@ -294,4 +308,29 @@ class MyPreferenceFragment : PreferenceFragmentCompat(), OnSharedPreferenceChang updatePrefSummary(p) } } + + // We use Preference and custom editor instead of EditTextPreference + // to hash password while it is saved and never have to show it, even hashed + + override fun onPreferenceTreeClick(preference: Preference?): Boolean { + if (preference != null) { + if (preference.key == resourceHelper.gs(R.string.key_master_password)) { + passwordCheck.setPassword(this.context!!, R.string.master_password, R.string.key_master_password) + return true; + } + if (preference.key == resourceHelper.gs(R.string.key_settings_password)) { + passwordCheck.setPassword(this.context!!, R.string.settings_password, R.string.key_settings_password) + return true; + } + if (preference.key == resourceHelper.gs(R.string.key_bolus_password)) { + passwordCheck.setPassword(this.context!!, R.string.bolus_password, R.string.key_bolus_password) + return true; + } + if (preference.key == resourceHelper.gs(R.string.key_application_password)) { + passwordCheck.setPassword(this.context!!, R.string.application_password, R.string.key_application_password) + return true; + } + } + return super.onPreferenceTreeClick(preference) + } } \ No newline at end of file diff --git a/app/src/main/java/info/nightscout/androidaps/utils/CryptoUtil.kt b/app/src/main/java/info/nightscout/androidaps/utils/CryptoUtil.kt new file mode 100644 index 0000000000..745f4449ee --- /dev/null +++ b/app/src/main/java/info/nightscout/androidaps/utils/CryptoUtil.kt @@ -0,0 +1,142 @@ +package info.nightscout.androidaps.utils + +import org.spongycastle.util.encoders.Base64 +import java.nio.ByteBuffer +import java.security.MessageDigest +import java.security.SecureRandom +import java.security.spec.KeySpec +import javax.crypto.Cipher +import javax.crypto.Mac +import javax.crypto.SecretKey +import javax.crypto.SecretKeyFactory +import javax.crypto.spec.GCMParameterSpec +import javax.crypto.spec.PBEKeySpec +import javax.crypto.spec.SecretKeySpec + +private val HEX_CHARS = "0123456789abcdef" +private val HEX_CHARS_ARRAY = "0123456789abcdef".toCharArray() + +fun String.hexStringToByteArray() : ByteArray { + + val upperCased = this.toLowerCase() + val result = ByteArray(length / 2) + for (i in 0 until length step 2) { + val firstIndex = HEX_CHARS.indexOf(upperCased[i]); + val secondIndex = HEX_CHARS.indexOf(upperCased[i + 1]); + + val octet = firstIndex.shl(4).or(secondIndex) + result.set(i.shr(1), octet.toByte()) + } + + return result +} + +fun ByteArray.toHex() : String{ + val result = StringBuffer() + + forEach { + val octet = it.toInt() + val firstIndex = (octet and 0xF0).ushr(4) + val secondIndex = octet and 0x0F + result.append(HEX_CHARS_ARRAY[firstIndex]) + result.append(HEX_CHARS_ARRAY[secondIndex]) + } + + return result.toString() +} + +object CryptoUtil { + + private const val IV_LENGTH_BYTE = 12 + private const val TAG_LENGTH_BIT = 128 + private const val AES_KEY_SIZE_BIT = 256 + private const val PBKDF2_ITERATIONS = 50000 // check delays it cause on real device + private const val SALT_SIZE_BYTE = 32 + + private val secureRandom: SecureRandom = SecureRandom() + + fun sha256(source: String): String { + val digest = MessageDigest.getInstance("SHA-256") + val hashRaw = digest.digest(source.toByteArray()) + return hashRaw.toHex() + } + + fun hmac256(str: String, secret: String): String? { + val sha256_HMAC = Mac.getInstance("HmacSHA256") + val secretKey = SecretKeySpec(secret.toByteArray(), "HmacSHA256") + sha256_HMAC.init(secretKey) + return sha256_HMAC.doFinal(str.toByteArray()).toHex() + } + + private fun prepCipherKey(passPhrase: String, salt:ByteArray, iterationCount:Int = PBKDF2_ITERATIONS, keyStrength:Int = AES_KEY_SIZE_BIT): SecretKeySpec { + val factory: SecretKeyFactory = SecretKeyFactory.getInstance("PBKDF2withHmacSHA1") + val spec: KeySpec = PBEKeySpec(passPhrase.toCharArray(), salt, iterationCount, keyStrength) + val tmp: SecretKey = factory.generateSecret(spec) + return SecretKeySpec(tmp.getEncoded(), "AES") + } + + fun mineSalt(len :Int = SALT_SIZE_BYTE): ByteArray { + val salt = ByteArray(len) + secureRandom.nextBytes(salt) + return salt + } + + fun encrypt(passPhrase: String, salt:ByteArray, rawData: String ): String? { + val iv: ByteArray? + val encrypted: ByteArray? + return try { + iv = ByteArray(IV_LENGTH_BYTE) + secureRandom.nextBytes(iv) + val cipherEnc: Cipher = Cipher.getInstance("AES/GCM/NoPadding") + cipherEnc.init(Cipher.ENCRYPT_MODE, prepCipherKey(passPhrase, salt), GCMParameterSpec(TAG_LENGTH_BIT, iv)) + encrypted = cipherEnc.doFinal(rawData.toByteArray()) + val byteBuffer: ByteBuffer = ByteBuffer.allocate(1 + iv.size + encrypted.size) + byteBuffer.put(iv.size.toByte()) + byteBuffer.put(iv) + byteBuffer.put(encrypted) + String(Base64.encode(byteBuffer.array())) + } catch (e: Exception) { + null + } + } + + fun decrypt(passPhrase: String, salt:ByteArray, encryptedData: String): String? { + val iv: ByteArray? + val encrypted: ByteArray? + return try { + val byteBuffer = ByteBuffer.wrap(Base64.decode(encryptedData)) + val ivLength = byteBuffer.get().toInt() + iv = ByteArray(ivLength) + byteBuffer[iv] + encrypted = ByteArray(byteBuffer.remaining()) + byteBuffer[encrypted] + val cipherDec: Cipher = Cipher.getInstance("AES/GCM/NoPadding") + cipherDec.init(Cipher.DECRYPT_MODE, prepCipherKey(passPhrase, salt), GCMParameterSpec(TAG_LENGTH_BIT, iv)) + val dec = cipherDec.doFinal(encrypted) + String(dec) + } catch (e: Exception) { + null + } + } + + fun checkPassword(password: String, referenceHash: String): Boolean { + return if (referenceHash.startsWith("hmac:")) { + val hashSegments = referenceHash.split(":") + if (hashSegments.size != 3) + return false + return hmac256(password, hashSegments[1]) == hashSegments[2] + } else { + password == referenceHash + } + } + + fun hashPassword(password: String): String { + return if (!password.startsWith("hmac:")) { + val salt = mineSalt().toHex() + return "hmac:${salt}:${hmac256(password, salt)}" + } else { + password + } + } + +} \ No newline at end of file diff --git a/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt b/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt index 6a2e18c732..4d1adfb233 100644 --- a/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt +++ b/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt @@ -2,13 +2,17 @@ package info.nightscout.androidaps.utils.protection import android.annotation.SuppressLint import android.app.AlertDialog +import android.content.Context import android.view.LayoutInflater import android.view.View import android.widget.EditText +import android.widget.ImageView import android.widget.TextView import androidx.annotation.StringRes +import androidx.appcompat.view.ContextThemeWrapper import androidx.fragment.app.FragmentActivity import info.nightscout.androidaps.R +import info.nightscout.androidaps.utils.CryptoUtil import info.nightscout.androidaps.utils.ToastUtils import info.nightscout.androidaps.utils.sharedPreferences.SP import javax.inject.Inject @@ -18,34 +22,74 @@ import javax.inject.Singleton class PasswordCheck @Inject constructor(val sp: SP) { @SuppressLint("InflateParams") - fun queryPassword(activity: FragmentActivity, @StringRes labelId: Int, @StringRes preference: Int, ok: Runnable?, cancel: Runnable? = null, fail: Runnable? = null) { + fun queryPassword(activity: FragmentActivity, @StringRes labelId: Int, @StringRes preference: Int, ok: ( (String) -> Unit)?, cancel: (()->Unit)? = null, fail: (()->Unit)? = null) { val password = sp.getString(preference, "") if (password == "") { - ok?.run() + ok?.invoke("") return } + + val titleLayout = activity.layoutInflater.inflate(R.layout.dialog_alert_custom, null) + (titleLayout.findViewById(R.id.alertdialog_title) as TextView).text = activity.getString(labelId) + (titleLayout.findViewById(R.id.alertdialog_icon) as ImageView).setImageResource(R.drawable.ic_key_48dp) + val promptsView = LayoutInflater.from(activity).inflate(R.layout.passwordprompt, null) - val alertDialogBuilder = AlertDialog.Builder(activity) + val alertDialogBuilder = AlertDialog.Builder(ContextThemeWrapper(activity, R.style.AppTheme)) alertDialogBuilder.setView(promptsView) - val label = promptsView.findViewById(R.id.passwordprompt_text) as TextView - label.text = activity.getString(labelId) val userInput = promptsView.findViewById(R.id.passwordprompt_pass) as EditText alertDialogBuilder .setCancelable(false) + .setCustomTitle(titleLayout) .setPositiveButton(activity.getString(R.string.ok)) { _, _ -> val enteredPassword = userInput.text.toString() - if (password == enteredPassword) ok?.run() + if (CryptoUtil.checkPassword(enteredPassword, password)) ok?.invoke(enteredPassword) else { ToastUtils.showToastInUiThread(activity, activity.getString(R.string.wrongpassword)) - fail?.run() + fail?.invoke() } } .setNegativeButton(activity.getString(R.string.cancel) ) { dialog, _ -> - cancel?.run() + cancel?.invoke() + dialog.cancel() + } + + alertDialogBuilder.create().show() + } + + @SuppressLint("InflateParams") + fun setPassword(context: Context, @StringRes labelId: Int, @StringRes preference: Int) { + val promptsView = LayoutInflater.from(context).inflate(R.layout.passwordprompt, null) + + val titleLayout = LayoutInflater.from(context).inflate(R.layout.dialog_alert_custom, null) + (titleLayout.findViewById(R.id.alertdialog_title) as TextView).text = context.getText(labelId) + (titleLayout.findViewById(R.id.alertdialog_icon) as ImageView).setImageResource(R.drawable.ic_key_48dp) + + val alertDialogBuilder = AlertDialog.Builder(ContextThemeWrapper(context, R.style.AppTheme)) + alertDialogBuilder.setView(promptsView) + + val userInput = promptsView.findViewById(R.id.passwordprompt_pass) as EditText + + alertDialogBuilder + .setCancelable(false) + .setCustomTitle(titleLayout) + .setPositiveButton(context.getString(R.string.ok)) { _, _ -> + val enteredPassword = userInput.text.toString() + if (enteredPassword.isNotEmpty()) { + sp.putString(preference, CryptoUtil.hashPassword(enteredPassword)) + } else { + if (sp.contains(preference)) { + sp.remove(preference) + } + } + ToastUtils.showToastInUiThread(context, context.getString(R.string.password_set)) + } + .setNegativeButton(context.getString(R.string.cancel) + ) { dialog, _ -> + ToastUtils.showToastInUiThread(context, context.getString(R.string.password_not_changed)) dialog.cancel() } diff --git a/app/src/main/java/info/nightscout/androidaps/utils/protection/ProtectionCheck.kt b/app/src/main/java/info/nightscout/androidaps/utils/protection/ProtectionCheck.kt index c4960f4999..a1cca9116e 100644 --- a/app/src/main/java/info/nightscout/androidaps/utils/protection/ProtectionCheck.kt +++ b/app/src/main/java/info/nightscout/androidaps/utils/protection/ProtectionCheck.kt @@ -21,7 +21,8 @@ class ProtectionCheck @Inject constructor( enum class ProtectionType { NONE, BIOMETRIC, - PASSWORD + MASTER_PASSWORD, + CUSTOM_PASSWORD } private val passwordsResourceIDs = listOf( @@ -43,7 +44,8 @@ class ProtectionCheck @Inject constructor( return when (ProtectionType.values()[sp.getInt(protectionTypeResourceIDs[protection.ordinal], ProtectionType.NONE.ordinal)]) { ProtectionType.NONE -> false ProtectionType.BIOMETRIC -> true - ProtectionType.PASSWORD -> sp.getString(passwordsResourceIDs[protection.ordinal], "") != "" + ProtectionType.MASTER_PASSWORD -> sp.getString(R.string.key_master_password, "") != "" + ProtectionType.CUSTOM_PASSWORD -> sp.getString(passwordsResourceIDs[protection.ordinal], "") != "" } } @@ -55,8 +57,10 @@ class ProtectionCheck @Inject constructor( ok?.run() ProtectionType.BIOMETRIC -> BiometricCheck.biometricPrompt(activity, titleResourceIDs[protection.ordinal], ok, cancel, fail) - ProtectionType.PASSWORD -> - passwordCheck.queryPassword(activity, titleResourceIDs[protection.ordinal], passwordsResourceIDs[protection.ordinal], ok, cancel, fail) + ProtectionType.MASTER_PASSWORD -> + passwordCheck.queryPassword(activity, R.string.master_password, R.string.key_master_password, { ok?.run() }, { cancel?.run() }, { fail?.run() }) + ProtectionType.CUSTOM_PASSWORD -> + passwordCheck.queryPassword(activity, titleResourceIDs[protection.ordinal], passwordsResourceIDs[protection.ordinal], { ok?.run() }, { cancel?.run() }, { fail?.run() }) } } } \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_key.xml b/app/src/main/res/drawable/ic_key.xml new file mode 100644 index 0000000000..4d4a6fa28d --- /dev/null +++ b/app/src/main/res/drawable/ic_key.xml @@ -0,0 +1,9 @@ + + + \ No newline at end of file diff --git a/app/src/main/res/drawable/ic_key_48dp.xml b/app/src/main/res/drawable/ic_key_48dp.xml new file mode 100644 index 0000000000..813dc24d00 --- /dev/null +++ b/app/src/main/res/drawable/ic_key_48dp.xml @@ -0,0 +1,17 @@ + + + + + \ No newline at end of file diff --git a/app/src/main/res/layout/passwordprompt.xml b/app/src/main/res/layout/passwordprompt.xml index 63ff30e531..ed2298e4d8 100644 --- a/app/src/main/res/layout/passwordprompt.xml +++ b/app/src/main/res/layout/passwordprompt.xml @@ -5,17 +5,13 @@ android:orientation="vertical" android:padding="10dp" > - - + android:hint="@string/password_hint" + android:inputType="textPassword" + > diff --git a/app/src/main/res/values/protection.xml b/app/src/main/res/values/protection.xml index 6bf675ab44..8b025f68cb 100644 --- a/app/src/main/res/values/protection.xml +++ b/app/src/main/res/values/protection.xml @@ -5,15 +5,22 @@ Settings protection Application protection Bolus protection + Master password Settings password Application password Bolus password Unlock settings Biometric - Password + Custom password No protection Protection + Password set! + Password not set + Password not changed + Enter password here + + master_password settings_password application_password translatable="false"bolus_password @@ -24,13 +31,15 @@ @string/noprotection @string/biometric - @string/password + @string/master_password + @string/custom_password 0 1 2 + 3 diff --git a/app/src/main/res/xml/pref_general.xml b/app/src/main/res/xml/pref_general.xml index 6daf95f3f8..b5f7074c7e 100644 --- a/app/src/main/res/xml/pref_general.xml +++ b/app/src/main/res/xml/pref_general.xml @@ -22,6 +22,12 @@ + + - @@ -41,7 +47,7 @@ android:key="@string/key_application_protection" android:title="@string/application_protection" /> - @@ -53,7 +59,7 @@ android:key="@string/key_bolus_protection" android:title="@string/bolus_protection" /> - diff --git a/app/src/test/java/info/nightscout/androidaps/utils/CryptoUtilTest.kt b/app/src/test/java/info/nightscout/androidaps/utils/CryptoUtilTest.kt new file mode 100644 index 0000000000..c147c0810c --- /dev/null +++ b/app/src/test/java/info/nightscout/androidaps/utils/CryptoUtilTest.kt @@ -0,0 +1,76 @@ +package info.nightscout.androidaps.utils + +import info.nightscout.androidaps.TestBase +import org.junit.Assert +import org.junit.Test +import org.junit.runner.RunWith +import org.powermock.core.classloader.annotations.PowerMockIgnore +import org.powermock.modules.junit4.PowerMockRunner + +@PowerMockIgnore("javax.crypto.*") +@RunWith(PowerMockRunner::class) +class CryptoUtilTest: TestBase() { + + @Test + fun testFixedSaltCrypto() { + val salt = byteArrayOf( + -33, -29, 16, -19, 99, -111, -3, 2, 116, 106, 47, 38, -54, 11, -77, 28, + 111, -15, -65, -110, 4, -32, -29, -70, -95, -88, -53, 19, 87, -103, 123, -15) + + val password = "thisIsFixedPassword" + val payload = "FIXED-PAYLOAD" + val encrypted = CryptoUtil.encrypt(password, salt, payload) + + Assert.assertNotNull(encrypted) + val decrypted = CryptoUtil.decrypt(password, salt, encrypted!!) + Assert.assertEquals(decrypted, payload) + } + + @Test + fun testStandardCrypto() { + val salt = CryptoUtil.mineSalt() + + val password = "topSikret" + val payload = "{what:payloadYouWantToProtect}" + val encrypted = CryptoUtil.encrypt(password, salt, payload) + + Assert.assertNotNull(encrypted) + val decrypted = CryptoUtil.decrypt(password, salt, encrypted!!) + Assert.assertEquals(decrypted, payload) + } + + @Test + fun testHashVector() { + val payload = "{what:payloadYouWantToProtect}" + val hash = CryptoUtil.sha256(payload) + Assert.assertEquals(hash, "a1aafe3ed6cc127e6d102ddbc40a205147230e9cfd178daf108c83543bbdcd13") + } + + @Test + fun testHmac() { + val payload = "{what:payloadYouWantToProtect}" + val password = "topSikret" + val expectedHmac = "ea2213953d0f2e55047cae2d23fb4f0de1b805d55e6271efa70d6b85fb692bea" // generated using other HMAC tool + val hash = CryptoUtil.hmac256(payload, password) + Assert.assertEquals(hash, expectedHmac) + } + + @Test + fun testPlainPasswordCheck() { + Assert.assertTrue(CryptoUtil.checkPassword("same", "same")) + Assert.assertFalse(CryptoUtil.checkPassword("same", "other")) + } + + @Test + fun testHashedPasswordCheck() { + Assert.assertTrue(CryptoUtil.checkPassword("givenSecret", CryptoUtil.hashPassword("givenSecret"))) + Assert.assertFalse(CryptoUtil.checkPassword("givenSecret", CryptoUtil.hashPassword("otherSecret"))) + + Assert.assertTrue(CryptoUtil.checkPassword("givenHashToCheck", "hmac:7fe5f9c7b4b97c5d32d5cfad9d07473543a9938dc07af48a46dbbb49f4f68c12:a0c7cee14312bbe31b51359a67f0d2dfdf46813f319180269796f1f617a64be1")) + Assert.assertFalse(CryptoUtil.checkPassword("givenMashToCheck", "hmac:7fe5f9c7b4b97c5d32d5cfad9d07473543a9938dc07af48a46dbbb49f4f68c12:a0c7cee14312bbe31b51359a67f0d2dfdf46813f319180269796f1f617a64be1")) + Assert.assertFalse(CryptoUtil.checkPassword("givenHashToCheck", "hmac:0fe5f9c7b4b97c5d32d5cfad9d07473543a9938dc07af48a46dbbb49f4f68c12:a0c7cee14312bbe31b51359a67f0d2dfdf46813f319180269796f1f617a64be1")) + Assert.assertFalse(CryptoUtil.checkPassword("givenHashToCheck", "hmac:7fe5f9c7b4b97c5d32d5cfad9d07473543a9938dc07af48a46dbbb49f4f68c12:b0c7cee14312bbe31b51359a67f0d2dfdf46813f319180269796f1f617a64be1")) + } + +} + From 39c08a5a2b68de4c184e537d467f0d1c18f21be9 Mon Sep 17 00:00:00 2001 From: Dominik Dzienia Date: Wed, 25 Mar 2020 14:37:23 +0100 Subject: [PATCH 2/6] Configured Autofill hint for password managers Added missing callback on password prompt dialog (will be usable in near future) --- .../utils/protection/PasswordCheck.kt | 22 ++++++++++++++++++- 1 file changed, 21 insertions(+), 1 deletion(-) diff --git a/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt b/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt index 4d1adfb233..02afbea43e 100644 --- a/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt +++ b/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt @@ -3,6 +3,7 @@ package info.nightscout.androidaps.utils.protection import android.annotation.SuppressLint import android.app.AlertDialog import android.content.Context +import android.os.Build import android.view.LayoutInflater import android.view.View import android.widget.EditText @@ -18,6 +19,9 @@ import info.nightscout.androidaps.utils.sharedPreferences.SP import javax.inject.Inject import javax.inject.Singleton +// since androidx.autofill.HintConstants are not available +val AUTOFILL_HINT_NEW_PASSWORD = "newPassword" + @Singleton class PasswordCheck @Inject constructor(val sp: SP) { @@ -39,6 +43,11 @@ class PasswordCheck @Inject constructor(val sp: SP) { alertDialogBuilder.setView(promptsView) val userInput = promptsView.findViewById(R.id.passwordprompt_pass) as EditText + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { + val autoFillHintPasswordKind = activity.getString(preference) + userInput.setAutofillHints(View.AUTOFILL_HINT_PASSWORD, "aaps_${autoFillHintPasswordKind}") + userInput.importantForAutofill = View.IMPORTANT_FOR_AUTOFILL_YES + } alertDialogBuilder .setCancelable(false) @@ -61,7 +70,7 @@ class PasswordCheck @Inject constructor(val sp: SP) { } @SuppressLint("InflateParams") - fun setPassword(context: Context, @StringRes labelId: Int, @StringRes preference: Int) { + fun setPassword(context: Context, @StringRes labelId: Int, @StringRes preference: Int, ok: ( (String) -> Unit)? = null, cancel: (()->Unit)? = null, clear: (()->Unit)? = null) { val promptsView = LayoutInflater.from(context).inflate(R.layout.passwordprompt, null) val titleLayout = LayoutInflater.from(context).inflate(R.layout.dialog_alert_custom, null) @@ -73,6 +82,12 @@ class PasswordCheck @Inject constructor(val sp: SP) { val userInput = promptsView.findViewById(R.id.passwordprompt_pass) as EditText + if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { + val autoFillHintPasswordKind = context.getString(preference) + userInput.setAutofillHints(AUTOFILL_HINT_NEW_PASSWORD, "aaps_${autoFillHintPasswordKind}") + userInput.importantForAutofill = View.IMPORTANT_FOR_AUTOFILL_YES + } + alertDialogBuilder .setCancelable(false) .setCustomTitle(titleLayout) @@ -80,9 +95,13 @@ class PasswordCheck @Inject constructor(val sp: SP) { val enteredPassword = userInput.text.toString() if (enteredPassword.isNotEmpty()) { sp.putString(preference, CryptoUtil.hashPassword(enteredPassword)) + ok?.invoke(enteredPassword) } else { if (sp.contains(preference)) { sp.remove(preference) + clear?.invoke() + } else { + cancel?.invoke() } } ToastUtils.showToastInUiThread(context, context.getString(R.string.password_set)) @@ -90,6 +109,7 @@ class PasswordCheck @Inject constructor(val sp: SP) { .setNegativeButton(context.getString(R.string.cancel) ) { dialog, _ -> ToastUtils.showToastInUiThread(context, context.getString(R.string.password_not_changed)) + cancel?.invoke() dialog.cancel() } From c58c62d4bcdc3ff87713b0c33c4f79e4a4a47317 Mon Sep 17 00:00:00 2001 From: Dominik Dzienia Date: Wed, 25 Mar 2020 14:47:28 +0100 Subject: [PATCH 3/6] Tost shown when password is set now show when password is not set at end (cancel) or unset --- .../nightscout/androidaps/utils/protection/PasswordCheck.kt | 5 ++++- app/src/main/res/values/protection.xml | 1 + 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt b/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt index 02afbea43e..65f1e76e44 100644 --- a/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt +++ b/app/src/main/java/info/nightscout/androidaps/utils/protection/PasswordCheck.kt @@ -95,16 +95,19 @@ class PasswordCheck @Inject constructor(val sp: SP) { val enteredPassword = userInput.text.toString() if (enteredPassword.isNotEmpty()) { sp.putString(preference, CryptoUtil.hashPassword(enteredPassword)) + ToastUtils.showToastInUiThread(context, context.getString(R.string.password_set)) ok?.invoke(enteredPassword) } else { if (sp.contains(preference)) { sp.remove(preference) + ToastUtils.showToastInUiThread(context, context.getString(R.string.password_cleared)) clear?.invoke() } else { + ToastUtils.showToastInUiThread(context, context.getString(R.string.password_not_changed)) cancel?.invoke() } } - ToastUtils.showToastInUiThread(context, context.getString(R.string.password_set)) + } .setNegativeButton(context.getString(R.string.cancel) ) { dialog, _ -> diff --git a/app/src/main/res/values/protection.xml b/app/src/main/res/values/protection.xml index 8b025f68cb..b044b9628e 100644 --- a/app/src/main/res/values/protection.xml +++ b/app/src/main/res/values/protection.xml @@ -18,6 +18,7 @@ Password set! Password not set Password not changed + Password cleared! Enter password here master_password From 3617885819e9ffc1321c8d92fa4ee985b2ef065b Mon Sep 17 00:00:00 2001 From: Milos Kozak Date: Thu, 26 Mar 2020 22:43:49 +0100 Subject: [PATCH 4/6] Fix WearFragment --- .../androidaps/dependencyInjection/FragmentsModule.kt | 2 ++ 1 file changed, 2 insertions(+) diff --git a/app/src/main/java/info/nightscout/androidaps/dependencyInjection/FragmentsModule.kt b/app/src/main/java/info/nightscout/androidaps/dependencyInjection/FragmentsModule.kt index dbd9cdf424..63ed16fb4c 100644 --- a/app/src/main/java/info/nightscout/androidaps/dependencyInjection/FragmentsModule.kt +++ b/app/src/main/java/info/nightscout/androidaps/dependencyInjection/FragmentsModule.kt @@ -28,6 +28,7 @@ import info.nightscout.androidaps.plugins.general.overview.OverviewFragment import info.nightscout.androidaps.plugins.general.overview.dialogs.EditQuickWizardDialog import info.nightscout.androidaps.plugins.general.smsCommunicator.SmsCommunicatorFragment import info.nightscout.androidaps.plugins.general.tidepool.TidepoolFragment +import info.nightscout.androidaps.plugins.general.wear.WearFragment import info.nightscout.androidaps.plugins.insulin.InsulinFragment import info.nightscout.androidaps.plugins.profile.local.LocalProfileFragment import info.nightscout.androidaps.plugins.profile.ns.NSProfileFragment @@ -71,6 +72,7 @@ abstract class FragmentsModule { @ContributesAndroidInjector abstract fun contributesNSProfileFragment(): NSProfileFragment @ContributesAndroidInjector abstract fun contributesNSClientFragment(): NSClientFragment @ContributesAndroidInjector abstract fun contributesSmsCommunicatorFragment(): SmsCommunicatorFragment + @ContributesAndroidInjector abstract fun contributesWearFragment(): WearFragment @ContributesAndroidInjector abstract fun contributesTidepoolFragment(): TidepoolFragment @ContributesAndroidInjector abstract fun contributesTreatmentsFragment(): TreatmentsFragment From 7e61c5bc61ac0f51f3057aea02cd32e096e2a28a Mon Sep 17 00:00:00 2001 From: Milos Kozak Date: Fri, 27 Mar 2020 20:29:36 +0100 Subject: [PATCH 5/6] New Crowdin translations (#2530) * New translations objectives.xml (French) * New translations objectives.xml (Swedish) * New translations strings.xml (Swedish) * New translations strings.xml (Czech) --- app/src/main/res/values-cs-rCZ/strings.xml | 2 +- app/src/main/res/values-fr-rFR/objectives.xml | 1 + app/src/main/res/values-sv-rSE/objectives.xml | 1 + app/src/main/res/values-sv-rSE/strings.xml | 1 + 4 files changed, 4 insertions(+), 1 deletion(-) diff --git a/app/src/main/res/values-cs-rCZ/strings.xml b/app/src/main/res/values-cs-rCZ/strings.xml index b89be5648a..d879db8f49 100644 --- a/app/src/main/res/values-cs-rCZ/strings.xml +++ b/app/src/main/res/values-cs-rCZ/strings.xml @@ -280,7 +280,7 @@ DanaR profil DIA [h] Celková doba aktivity inzulínu - Chyba při nastavení dočasného bazálu + Chyba při nastavení bazálního pprofilu Načíst Nahrávám E bolus diff --git a/app/src/main/res/values-fr-rFR/objectives.xml b/app/src/main/res/values-fr-rFR/objectives.xml index 37d434b2d3..58f83d4a9c 100644 --- a/app/src/main/res/values-fr-rFR/objectives.xml +++ b/app/src/main/res/values-fr-rFR/objectives.xml @@ -35,6 +35,7 @@ Affichage du contenu du plugin Boucle Modification de l\'échelle du graphique par un appui long sur la courbe de glycémie Entrer + Si vous avez au moins 3 mois d\'expérience de boucle fermée avec d\'autres systèmes, vous pourriez avoir droit à un code permettant d\'ignorer les objectifs. Voir https://androidaps.readthedocs.io/en/latest/CROWDIN/fr/Usage/Objectives.html#ignorer-les-objectifs pour plus de détails. Code accepté Code invalide Prouver ses connaissances diff --git a/app/src/main/res/values-sv-rSE/objectives.xml b/app/src/main/res/values-sv-rSE/objectives.xml index f9c5cd13ff..94b2ba5549 100644 --- a/app/src/main/res/values-sv-rSE/objectives.xml +++ b/app/src/main/res/values-sv-rSE/objectives.xml @@ -35,6 +35,7 @@ Visa innehållet i insticksprogrammet \"Loop\" Testa skala om BG-grafen genom att trycka och hålla in fingret på den Enter + Om du har minst 3 månaders erfarenhet av closed loop med andra system kan du kvalificera dig för en kod för att hoppa över mål. Se https://androidaps.readthedocs.io/en/latest/EN/Usage/Objectives.html#skip-objectives för mer info. Koden godkänd Koden är felaktig Bevisa dina kunskaper diff --git a/app/src/main/res/values-sv-rSE/strings.xml b/app/src/main/res/values-sv-rSE/strings.xml index 7e1ec0c8ed..4fada83e89 100644 --- a/app/src/main/res/values-sv-rSE/strings.xml +++ b/app/src/main/res/values-sv-rSE/strings.xml @@ -1454,4 +1454,5 @@ Eversense-appen. SMB utförd Basalförändring begärd Basalförändring utförd + Pumpvarningar Insight From c4776492768d80daacb15fdbfa0a49adaa6c50fe Mon Sep 17 00:00:00 2001 From: Milos Kozak Date: Fri, 27 Mar 2020 21:32:47 +0100 Subject: [PATCH 6/6] Fix wear preferences, expand single preference by default --- .../activities/MyPreferenceFragment.kt | 54 ++++++++++--------- app/src/main/res/xml/pref_wear.xml | 9 ++-- 2 files changed, 33 insertions(+), 30 deletions(-) diff --git a/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt b/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt index 122ec8d5f0..a7e15ea2fa 100644 --- a/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt +++ b/app/src/main/java/info/nightscout/androidaps/activities/MyPreferenceFragment.kt @@ -50,7 +50,6 @@ import info.nightscout.androidaps.plugins.source.EversensePlugin import info.nightscout.androidaps.plugins.source.GlimpPlugin import info.nightscout.androidaps.plugins.source.PoctechPlugin import info.nightscout.androidaps.plugins.source.TomatoPlugin -import info.nightscout.androidaps.utils.CryptoUtil import info.nightscout.androidaps.utils.OKDialog.show import info.nightscout.androidaps.utils.SafeParse import info.nightscout.androidaps.utils.protection.PasswordCheck @@ -189,7 +188,7 @@ class MyPreferenceFragment : PreferenceFragmentCompat(), OnSharedPreferenceChang addPreferencesFromResource(R.xml.pref_datachoices, rootKey) addPreferencesFromResourceIfEnabled(maintenancePlugin, rootKey) } - initSummary(preferenceScreen) + initSummary(preferenceScreen, pluginId != -1) for (plugin in pluginStore.plugins) { plugin.preprocessPreferences(this) } @@ -287,22 +286,27 @@ class MyPreferenceFragment : PreferenceFragmentCompat(), OnSharedPreferenceChang } if (pref is Preference) { - if ((pref.getKey() != null) && (pref.getKey().contains("_password"))) { - if (sp.getString(pref.getKey(), "").startsWith("hmac:")) { - pref.setSummary("******") + if ((pref.key != null) && (pref.key.contains("_password"))) { + if (sp.getString(pref.key, "").startsWith("hmac:")) { + pref.summary = "******" } else { - pref.setSummary(resourceHelper.gs(R.string.password_not_set)) + pref.summary = resourceHelper.gs(R.string.password_not_set) } } } pref?.let { adjustUnitDependentPrefs(it) } } - private fun initSummary(p: Preference) { + private fun initSummary(p: Preference, isSinglePreference: Boolean) { p.isIconSpaceReserved = false // remove extra spacing on left after migration to androidx + // expand single plugin preference by default + if (p is PreferenceScreen && isSinglePreference) { + if (p.size > 0 && p.getPreference(0) is PreferenceCategory) + (p.getPreference(0) as PreferenceCategory).initialExpandedChildrenCount = Int.MAX_VALUE + } if (p is PreferenceGroup) { for (i in 0 until p.preferenceCount) { - initSummary(p.getPreference(i)) + initSummary(p.getPreference(i), isSinglePreference) } } else { updatePrefSummary(p) @@ -313,22 +317,24 @@ class MyPreferenceFragment : PreferenceFragmentCompat(), OnSharedPreferenceChang // to hash password while it is saved and never have to show it, even hashed override fun onPreferenceTreeClick(preference: Preference?): Boolean { - if (preference != null) { - if (preference.key == resourceHelper.gs(R.string.key_master_password)) { - passwordCheck.setPassword(this.context!!, R.string.master_password, R.string.key_master_password) - return true; - } - if (preference.key == resourceHelper.gs(R.string.key_settings_password)) { - passwordCheck.setPassword(this.context!!, R.string.settings_password, R.string.key_settings_password) - return true; - } - if (preference.key == resourceHelper.gs(R.string.key_bolus_password)) { - passwordCheck.setPassword(this.context!!, R.string.bolus_password, R.string.key_bolus_password) - return true; - } - if (preference.key == resourceHelper.gs(R.string.key_application_password)) { - passwordCheck.setPassword(this.context!!, R.string.application_password, R.string.key_application_password) - return true; + context?.let { context -> + if (preference != null) { + if (preference.key == resourceHelper.gs(R.string.key_master_password)) { + passwordCheck.setPassword(context, R.string.master_password, R.string.key_master_password) + return true + } + if (preference.key == resourceHelper.gs(R.string.key_settings_password)) { + passwordCheck.setPassword(context, R.string.settings_password, R.string.key_settings_password) + return true + } + if (preference.key == resourceHelper.gs(R.string.key_bolus_password)) { + passwordCheck.setPassword(context, R.string.bolus_password, R.string.key_bolus_password) + return true + } + if (preference.key == resourceHelper.gs(R.string.key_application_password)) { + passwordCheck.setPassword(context, R.string.application_password, R.string.key_application_password) + return true + } } } return super.onPreferenceTreeClick(preference) diff --git a/app/src/main/res/xml/pref_wear.xml b/app/src/main/res/xml/pref_wear.xml index a18d77061b..855ae4a9af 100644 --- a/app/src/main/res/xml/pref_wear.xml +++ b/app/src/main/res/xml/pref_wear.xml @@ -15,8 +15,7 @@ + android:title="@string/wear_wizard_settings"> + android:title="@string/wear_display_settings"> + android:title="@string/wear_general_settings">