fix: Separate Main credential, remove ContentProvider from Database

This commit is contained in:
J-Jamet 2023-05-14 15:33:39 +02:00
parent 7db0c4f7d3
commit 6742a8893c
25 changed files with 339 additions and 279 deletions

View file

@ -54,7 +54,7 @@ import com.kunzisoft.keepass.app.database.FileDatabaseHistoryAction
import com.kunzisoft.keepass.autofill.AutofillComponent import com.kunzisoft.keepass.autofill.AutofillComponent
import com.kunzisoft.keepass.autofill.AutofillHelper import com.kunzisoft.keepass.autofill.AutofillHelper
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.education.FileDatabaseSelectActivityEducation import com.kunzisoft.keepass.education.FileDatabaseSelectActivityEducation
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.model.RegisterInfo import com.kunzisoft.keepass.model.RegisterInfo

View file

@ -75,10 +75,10 @@ import com.kunzisoft.keepass.adapters.BreadcrumbAdapter
import com.kunzisoft.keepass.autofill.AutofillComponent import com.kunzisoft.keepass.autofill.AutofillComponent
import com.kunzisoft.keepass.autofill.AutofillHelper import com.kunzisoft.keepass.autofill.AutofillHelper
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.element.DateInstant import com.kunzisoft.keepass.database.element.DateInstant
import com.kunzisoft.keepass.database.element.Entry import com.kunzisoft.keepass.database.element.Entry
import com.kunzisoft.keepass.database.element.Group import com.kunzisoft.keepass.database.element.Group
import com.kunzisoft.keepass.database.element.MainCredential
import com.kunzisoft.keepass.database.element.SortNodeEnum import com.kunzisoft.keepass.database.element.SortNodeEnum
import com.kunzisoft.keepass.database.element.node.Node import com.kunzisoft.keepass.database.element.node.Node
import com.kunzisoft.keepass.database.element.node.NodeId import com.kunzisoft.keepass.database.element.node.NodeId

View file

@ -56,7 +56,7 @@ import com.kunzisoft.keepass.autofill.AutofillHelper
import com.kunzisoft.keepass.biometric.AdvancedUnlockFragment import com.kunzisoft.keepass.biometric.AdvancedUnlockFragment
import com.kunzisoft.keepass.biometric.AdvancedUnlockManager import com.kunzisoft.keepass.biometric.AdvancedUnlockManager
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.exception.DuplicateUuidDatabaseException import com.kunzisoft.keepass.database.exception.DuplicateUuidDatabaseException
import com.kunzisoft.keepass.database.exception.FileNotFoundDatabaseException import com.kunzisoft.keepass.database.exception.FileNotFoundDatabaseException
import com.kunzisoft.keepass.education.PasswordActivityEducation import com.kunzisoft.keepass.education.PasswordActivityEducation

View file

@ -27,7 +27,7 @@ import android.widget.TextView
import androidx.appcompat.app.AlertDialog import androidx.appcompat.app.AlertDialog
import com.kunzisoft.keepass.R import com.kunzisoft.keepass.R
import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.utils.UriUtil.getDocumentFile import com.kunzisoft.keepass.utils.UriUtil.getDocumentFile
import com.kunzisoft.keepass.view.MainCredentialView import com.kunzisoft.keepass.view.MainCredentialView

View file

@ -26,7 +26,7 @@ import android.os.Bundle
import androidx.appcompat.app.AlertDialog import androidx.appcompat.app.AlertDialog
import androidx.fragment.app.DialogFragment import androidx.fragment.app.DialogFragment
import com.kunzisoft.keepass.R import com.kunzisoft.keepass.R
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
class PasswordEncodingDialogFragment : DialogFragment() { class PasswordEncodingDialogFragment : DialogFragment() {
@ -78,8 +78,10 @@ class PasswordEncodingDialogFragment : DialogFragment() {
private const val DATABASE_URI_KEY = "DATABASE_URI_KEY" private const val DATABASE_URI_KEY = "DATABASE_URI_KEY"
private const val MAIN_CREDENTIAL = "MAIN_CREDENTIAL" private const val MAIN_CREDENTIAL = "MAIN_CREDENTIAL"
fun getInstance(databaseUri: Uri, fun getInstance(
mainCredential: MainCredential): SortDialogFragment { databaseUri: Uri,
mainCredential: MainCredential
): SortDialogFragment {
val fragment = SortDialogFragment() val fragment = SortDialogFragment()
fragment.arguments = Bundle().apply { fragment.arguments = Bundle().apply {
putParcelable(DATABASE_URI_KEY, databaseUri) putParcelable(DATABASE_URI_KEY, databaseUri)

View file

@ -35,7 +35,7 @@ import com.google.android.material.textfield.TextInputLayout
import com.kunzisoft.keepass.R import com.kunzisoft.keepass.R
import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper
import com.kunzisoft.keepass.activities.helpers.setOpenDocumentClickListener import com.kunzisoft.keepass.activities.helpers.setOpenDocumentClickListener
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.hardware.HardwareKeyActivity import com.kunzisoft.keepass.hardware.HardwareKeyActivity
import com.kunzisoft.keepass.password.PasswordEntropy import com.kunzisoft.keepass.password.PasswordEntropy

View file

@ -5,8 +5,8 @@ import android.os.Bundle
import androidx.activity.viewModels import androidx.activity.viewModels
import com.kunzisoft.keepass.activities.stylish.StylishActivity import com.kunzisoft.keepass.activities.stylish.StylishActivity
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.action.DatabaseTaskProvider import com.kunzisoft.keepass.database.action.DatabaseTaskProvider
import com.kunzisoft.keepass.database.element.MainCredential
import com.kunzisoft.keepass.model.CipherEncryptDatabase import com.kunzisoft.keepass.model.CipherEncryptDatabase
import com.kunzisoft.keepass.tasks.ActionRunnable import com.kunzisoft.keepass.tasks.ActionRunnable
import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir

View file

@ -37,10 +37,10 @@ import com.kunzisoft.keepass.activities.dialogs.PasswordEncodingDialogFragment
import com.kunzisoft.keepass.activities.helpers.EntrySelectionHelper import com.kunzisoft.keepass.activities.helpers.EntrySelectionHelper
import com.kunzisoft.keepass.activities.helpers.SpecialMode import com.kunzisoft.keepass.activities.helpers.SpecialMode
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.element.Database import com.kunzisoft.keepass.database.element.Database
import com.kunzisoft.keepass.database.element.Entry import com.kunzisoft.keepass.database.element.Entry
import com.kunzisoft.keepass.database.element.Group import com.kunzisoft.keepass.database.element.Group
import com.kunzisoft.keepass.database.element.MainCredential
import com.kunzisoft.keepass.database.element.node.Node import com.kunzisoft.keepass.database.element.node.Node
import com.kunzisoft.keepass.database.element.node.NodeId import com.kunzisoft.keepass.database.element.node.NodeId
import com.kunzisoft.keepass.model.GroupInfo import com.kunzisoft.keepass.model.GroupInfo
@ -251,7 +251,7 @@ abstract class DatabaseLockActivity : DatabaseModeActivity(),
mDatabase?.let { database -> mDatabase?.let { database ->
database.fileUri?.let { databaseUri -> database.fileUri?.let { databaseUri ->
// Show the progress dialog now or after dialog confirmation // Show the progress dialog now or after dialog confirmation
if (database.validatePasswordEncoding(mainCredential)) { if (database.isValidCredential(mainCredential.toMasterCredential(contentResolver))) {
assignDatabasePassword(databaseUri, mainCredential) assignDatabasePassword(databaseUri, mainCredential)
} else { } else {
PasswordEncodingDialogFragment.getInstance(databaseUri, mainCredential) PasswordEncodingDialogFragment.getInstance(databaseUri, mainCredential)

View file

@ -1,5 +1,6 @@
package com.kunzisoft.keepass.database package com.kunzisoft.keepass.database
import android.net.Uri
import com.kunzisoft.keepass.database.element.Database import com.kunzisoft.keepass.database.element.Database
import com.kunzisoft.keepass.database.element.icon.IconImageCustom import com.kunzisoft.keepass.database.element.icon.IconImageCustom
import com.kunzisoft.keepass.icons.IconDrawableFactory import com.kunzisoft.keepass.icons.IconDrawableFactory
@ -8,6 +9,8 @@ import java.io.File
class ContextualDatabase: Database() { class ContextualDatabase: Database() {
var fileUri: Uri? = null
val iconDrawableFactory = IconDrawableFactory( val iconDrawableFactory = IconDrawableFactory(
retrieveBinaryCache = { binaryCache }, retrieveBinaryCache = { binaryCache },
retrieveCustomIconBinary = { iconId -> getBinaryForCustomIcon(iconId) } retrieveCustomIconBinary = { iconId -> getBinaryForCustomIcon(iconId) }
@ -23,6 +26,11 @@ class ContextualDatabase: Database() {
super.clearIndexesAndBinaries(filesDirectory) super.clearIndexesAndBinaries(filesDirectory)
} }
override fun clearAndClose(filesDirectory: File?) {
super.clearAndClose(filesDirectory)
this.fileUri = null
}
companion object : SingletonHolder<ContextualDatabase>(::ContextualDatabase) { companion object : SingletonHolder<ContextualDatabase>(::ContextualDatabase) {
private val TAG = ContextualDatabase::class.java.name private val TAG = ContextualDatabase::class.java.name
} }

View file

@ -0,0 +1,100 @@
/*
* Copyright 2022 Jeremy Jamet / Kunzisoft.
*
* This file is part of KeePassDX.
*
* KeePassDX is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* KeePassDX is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with KeePassDX. If not, see <http://www.gnu.org/licenses/>.
*/
package com.kunzisoft.keepass.database
import android.content.ContentResolver
import android.net.Uri
import android.os.Parcel
import android.os.Parcelable
import com.kunzisoft.keepass.database.element.MasterCredential
import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.utils.UriHelper.getUriInputStream
import com.kunzisoft.keepass.utils.readEnum
import com.kunzisoft.keepass.utils.writeEnum
data class MainCredential(var password: String? = null,
var keyFileUri: Uri? = null,
var hardwareKey: HardwareKey? = null): Parcelable {
constructor(parcel: Parcel) : this() {
password = parcel.readString()
keyFileUri = parcel.readParcelable(Uri::class.java.classLoader)
hardwareKey = parcel.readEnum<HardwareKey>()
}
override fun writeToParcel(parcel: Parcel, flags: Int) {
parcel.writeString(password)
parcel.writeParcelable(keyFileUri, flags)
parcel.writeEnum(hardwareKey)
}
override fun describeContents(): Int {
return 0
}
override fun equals(other: Any?): Boolean {
if (this === other) return true
if (javaClass != other?.javaClass) return false
other as MainCredential
if (password != other.password) return false
if (keyFileUri != other.keyFileUri) return false
if (hardwareKey != other.hardwareKey) return false
return true
}
override fun hashCode(): Int {
var result = password?.hashCode() ?: 0
result = 31 * result + (keyFileUri?.hashCode() ?: 0)
result = 31 * result + (hardwareKey?.hashCode() ?: 0)
return result
}
fun toMasterCredential(contentResolver: ContentResolver): MasterCredential {
return MasterCredential(
this.password,
this.keyFileUri?.let {
getKeyFileData(contentResolver, it)
},
this.hardwareKey
)
}
private fun getKeyFileData(contentResolver: ContentResolver,
keyFileUri: Uri): ByteArray? {
contentResolver.getUriInputStream(keyFileUri)?.use { keyFileInputStream ->
return keyFileInputStream.readBytes()
}
return null
}
companion object CREATOR : Parcelable.Creator<MainCredential> {
override fun createFromParcel(parcel: Parcel): MainCredential {
return MainCredential(parcel)
}
override fun newArray(size: Int): Array<MainCredential?> {
return arrayOfNulls(size)
}
private val TAG = MainCredential::class.java.simpleName
}
}

View file

@ -24,7 +24,7 @@ import android.net.Uri
import com.kunzisoft.keepass.app.database.CipherDatabaseAction import com.kunzisoft.keepass.app.database.CipherDatabaseAction
import com.kunzisoft.keepass.app.database.FileDatabaseHistoryAction import com.kunzisoft.keepass.app.database.FileDatabaseHistoryAction
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
open class AssignMainCredentialInDatabaseRunnable ( open class AssignMainCredentialInDatabaseRunnable (

View file

@ -24,7 +24,7 @@ import android.net.Uri
import android.util.Log import android.util.Log
import com.kunzisoft.keepass.app.database.FileDatabaseHistoryAction import com.kunzisoft.keepass.app.database.FileDatabaseHistoryAction
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.settings.PreferencesUtil import com.kunzisoft.keepass.settings.PreferencesUtil
import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir
@ -45,7 +45,8 @@ class CreateDatabaseRunnable(
try { try {
// Create new database record // Create new database record
mDatabase.apply { mDatabase.apply {
createData(mDatabaseUri, databaseName, rootName, templateGroupName) this.fileUri = mDatabaseUri
createData(databaseName, rootName, templateGroupName)
} }
} catch (e: Exception) { } catch (e: Exception) {
mDatabase.clearAndClose(context.getBinaryDir()) mDatabase.clearAndClose(context.getBinaryDir())

View file

@ -40,11 +40,11 @@ import com.kunzisoft.keepass.R
import com.kunzisoft.keepass.activities.dialogs.DatabaseChangedDialogFragment import com.kunzisoft.keepass.activities.dialogs.DatabaseChangedDialogFragment
import com.kunzisoft.keepass.activities.dialogs.DatabaseChangedDialogFragment.Companion.DATABASE_CHANGED_DIALOG_TAG import com.kunzisoft.keepass.activities.dialogs.DatabaseChangedDialogFragment.Companion.DATABASE_CHANGED_DIALOG_TAG
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.crypto.EncryptionAlgorithm import com.kunzisoft.keepass.database.crypto.EncryptionAlgorithm
import com.kunzisoft.keepass.database.crypto.kdf.KdfEngine import com.kunzisoft.keepass.database.crypto.kdf.KdfEngine
import com.kunzisoft.keepass.database.element.Entry import com.kunzisoft.keepass.database.element.Entry
import com.kunzisoft.keepass.database.element.Group import com.kunzisoft.keepass.database.element.Group
import com.kunzisoft.keepass.database.element.MainCredential
import com.kunzisoft.keepass.database.element.database.CompressionAlgorithm import com.kunzisoft.keepass.database.element.database.CompressionAlgorithm
import com.kunzisoft.keepass.database.element.node.Node import com.kunzisoft.keepass.database.element.node.Node
import com.kunzisoft.keepass.database.element.node.NodeId import com.kunzisoft.keepass.database.element.node.NodeId

View file

@ -24,14 +24,16 @@ import android.net.Uri
import com.kunzisoft.keepass.app.database.CipherDatabaseAction import com.kunzisoft.keepass.app.database.CipherDatabaseAction
import com.kunzisoft.keepass.app.database.FileDatabaseHistoryAction import com.kunzisoft.keepass.app.database.FileDatabaseHistoryAction
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.element.binary.BinaryData import com.kunzisoft.keepass.database.element.binary.BinaryData
import com.kunzisoft.keepass.database.exception.DatabaseInputException import com.kunzisoft.keepass.database.exception.DatabaseInputException
import com.kunzisoft.keepass.database.exception.UnknownDatabaseLocationException
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.model.CipherEncryptDatabase import com.kunzisoft.keepass.model.CipherEncryptDatabase
import com.kunzisoft.keepass.settings.PreferencesUtil import com.kunzisoft.keepass.settings.PreferencesUtil
import com.kunzisoft.keepass.tasks.ActionRunnable import com.kunzisoft.keepass.tasks.ActionRunnable
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater import com.kunzisoft.keepass.tasks.ProgressTaskUpdater
import com.kunzisoft.keepass.utils.UriHelper.getUriInputStream
import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir
class LoadDatabaseRunnable( class LoadDatabaseRunnable(
@ -56,10 +58,13 @@ class LoadDatabaseRunnable(
override fun onActionRun() { override fun onActionRun() {
try { try {
val contentResolver = context.contentResolver
// Save database URI
mDatabase.fileUri = mDatabaseUri
mDatabase.loadData( mDatabase.loadData(
context.contentResolver, contentResolver.getUriInputStream(mDatabaseUri)
mDatabaseUri, ?: throw UnknownDatabaseLocationException(),
mMainCredential, mMainCredential.toMasterCredential(contentResolver),
mChallengeResponseRetriever, mChallengeResponseRetriever,
mReadonly, mReadonly,
binaryDir, binaryDir,

View file

@ -22,12 +22,14 @@ package com.kunzisoft.keepass.database.action
import android.content.Context import android.content.Context
import android.net.Uri import android.net.Uri
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.element.binary.BinaryData import com.kunzisoft.keepass.database.element.binary.BinaryData
import com.kunzisoft.keepass.database.exception.DatabaseException import com.kunzisoft.keepass.database.exception.DatabaseException
import com.kunzisoft.keepass.database.exception.UnknownDatabaseLocationException
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.settings.PreferencesUtil import com.kunzisoft.keepass.settings.PreferencesUtil
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater import com.kunzisoft.keepass.tasks.ProgressTaskUpdater
import com.kunzisoft.keepass.utils.UriHelper.getUriInputStream
class MergeDatabaseRunnable( class MergeDatabaseRunnable(
context: Context, context: Context,
@ -48,10 +50,12 @@ class MergeDatabaseRunnable(
override fun onActionRun() { override fun onActionRun() {
try { try {
val contentResolver = context.contentResolver
database.mergeData( database.mergeData(
context.contentResolver, context.contentResolver.getUriInputStream(
mDatabaseToMergeUri, mDatabaseToMergeUri ?: database.fileUri
mDatabaseToMergeMainCredential, ) ?: throw UnknownDatabaseLocationException(),
mDatabaseToMergeMainCredential?.toMasterCredential(contentResolver),
mDatabaseToMergeChallengeResponseRetriever, mDatabaseToMergeChallengeResponseRetriever,
{ memoryWanted -> { memoryWanted ->
BinaryData.canMemoryBeAllocatedInRAM(context, memoryWanted) BinaryData.canMemoryBeAllocatedInRAM(context, memoryWanted)

View file

@ -23,9 +23,11 @@ import android.content.Context
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.binary.BinaryData import com.kunzisoft.keepass.database.element.binary.BinaryData
import com.kunzisoft.keepass.database.exception.DatabaseException import com.kunzisoft.keepass.database.exception.DatabaseException
import com.kunzisoft.keepass.database.exception.UnknownDatabaseLocationException
import com.kunzisoft.keepass.settings.PreferencesUtil import com.kunzisoft.keepass.settings.PreferencesUtil
import com.kunzisoft.keepass.tasks.ActionRunnable import com.kunzisoft.keepass.tasks.ActionRunnable
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater import com.kunzisoft.keepass.tasks.ProgressTaskUpdater
import com.kunzisoft.keepass.utils.UriHelper.getUriInputStream
import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir import com.kunzisoft.keepass.utils.UriUtil.getBinaryDir
class ReloadDatabaseRunnable( class ReloadDatabaseRunnable(
@ -45,7 +47,9 @@ class ReloadDatabaseRunnable(
override fun onActionRun() { override fun onActionRun() {
try { try {
mDatabase.reloadData(context.contentResolver, mDatabase.reloadData(
context.contentResolver.getUriInputStream(mDatabase.fileUri)
?: throw UnknownDatabaseLocationException(),
{ memoryWanted -> { memoryWanted ->
BinaryData.canMemoryBeAllocatedInRAM(context, memoryWanted) BinaryData.canMemoryBeAllocatedInRAM(context, memoryWanted)
}, },

View file

@ -22,10 +22,12 @@ package com.kunzisoft.keepass.database.action
import android.content.Context import android.content.Context
import android.net.Uri import android.net.Uri
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.exception.DatabaseException import com.kunzisoft.keepass.database.exception.DatabaseException
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.tasks.ActionRunnable import com.kunzisoft.keepass.tasks.ActionRunnable
import com.kunzisoft.keepass.utils.UriHelper.getUriOutputStream
import java.io.File
open class SaveDatabaseRunnable( open class SaveDatabaseRunnable(
protected var context: Context, protected var context: Context,
@ -44,11 +46,14 @@ open class SaveDatabaseRunnable(
database.checkVersion() database.checkVersion()
if (saveDatabase && result.isSuccess) { if (saveDatabase && result.isSuccess) {
try { try {
val contentResolver = context.contentResolver
// Build temp database file to avoid file corruption if error
database.saveData( database.saveData(
context.contentResolver, cacheFile = File(context.cacheDir, databaseCopyUri.hashCode().toString()),
context.cacheDir, databaseOutputStream = context.contentResolver
databaseCopyUri, .getUriOutputStream(databaseCopyUri ?: database.fileUri),
mainCredential, isNewLocation = databaseCopyUri == null,
mainCredential?.toMasterCredential(contentResolver),
challengeResponseRetriever) challengeResponseRetriever)
} catch (e: DatabaseException) { } catch (e: DatabaseException) {
setError(e) setError(e)

View file

@ -32,6 +32,7 @@ import androidx.media.app.NotificationCompat
import com.kunzisoft.keepass.R import com.kunzisoft.keepass.R
import com.kunzisoft.keepass.activities.GroupActivity import com.kunzisoft.keepass.activities.GroupActivity
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.action.* import com.kunzisoft.keepass.database.action.*
import com.kunzisoft.keepass.database.action.history.DeleteEntryHistoryDatabaseRunnable import com.kunzisoft.keepass.database.action.history.DeleteEntryHistoryDatabaseRunnable
import com.kunzisoft.keepass.database.action.history.RestoreEntryHistoryDatabaseRunnable import com.kunzisoft.keepass.database.action.history.RestoreEntryHistoryDatabaseRunnable
@ -39,7 +40,6 @@ import com.kunzisoft.keepass.database.action.node.*
import com.kunzisoft.keepass.database.element.Database import com.kunzisoft.keepass.database.element.Database
import com.kunzisoft.keepass.database.element.Entry import com.kunzisoft.keepass.database.element.Entry
import com.kunzisoft.keepass.database.element.Group import com.kunzisoft.keepass.database.element.Group
import com.kunzisoft.keepass.database.element.MainCredential
import com.kunzisoft.keepass.database.element.database.CompressionAlgorithm import com.kunzisoft.keepass.database.element.database.CompressionAlgorithm
import com.kunzisoft.keepass.database.element.node.Node import com.kunzisoft.keepass.database.element.node.Node
import com.kunzisoft.keepass.database.element.node.NodeId import com.kunzisoft.keepass.database.element.node.NodeId

View file

@ -36,7 +36,7 @@ import com.kunzisoft.keepass.activities.dialogs.SetMainCredentialDialogFragment
import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper
import com.kunzisoft.keepass.activities.legacy.DatabaseLockActivity import com.kunzisoft.keepass.activities.legacy.DatabaseLockActivity
import com.kunzisoft.keepass.database.ContextualDatabase import com.kunzisoft.keepass.database.ContextualDatabase
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.tasks.ActionRunnable import com.kunzisoft.keepass.tasks.ActionRunnable
import com.kunzisoft.keepass.timeout.TimeoutHelper import com.kunzisoft.keepass.timeout.TimeoutHelper
import com.kunzisoft.keepass.view.showActionErrorIfNeeded import com.kunzisoft.keepass.view.showActionErrorIfNeeded

View file

@ -38,9 +38,9 @@ import androidx.appcompat.app.AppCompatActivity
import com.kunzisoft.keepass.R import com.kunzisoft.keepass.R
import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper import com.kunzisoft.keepass.activities.helpers.ExternalFileHelper
import com.kunzisoft.keepass.activities.helpers.setOpenDocumentClickListener import com.kunzisoft.keepass.activities.helpers.setOpenDocumentClickListener
import com.kunzisoft.keepass.model.CredentialStorage import com.kunzisoft.keepass.database.MainCredential
import com.kunzisoft.keepass.database.element.MainCredential
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.model.CredentialStorage
class MainCredentialView @JvmOverloads constructor(context: Context, class MainCredentialView @JvmOverloads constructor(context: Context,
attrs: AttributeSet? = null, attrs: AttributeSet? = null,

View file

@ -19,9 +19,7 @@
*/ */
package com.kunzisoft.keepass.database.element package com.kunzisoft.keepass.database.element
import android.content.ContentResolver
import android.graphics.Color import android.graphics.Color
import android.net.Uri
import android.util.Log import android.util.Log
import com.kunzisoft.androidclearchroma.ChromaUtil import com.kunzisoft.androidclearchroma.ChromaUtil
import com.kunzisoft.keepass.database.crypto.EncryptionAlgorithm import com.kunzisoft.keepass.database.crypto.EncryptionAlgorithm
@ -56,8 +54,6 @@ import com.kunzisoft.keepass.database.search.SearchParameters
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.tasks.ProgressTaskUpdater import com.kunzisoft.keepass.tasks.ProgressTaskUpdater
import com.kunzisoft.keepass.utils.* import com.kunzisoft.keepass.utils.*
import com.kunzisoft.keepass.utils.UriHelper.getUriInputStream
import com.kunzisoft.keepass.utils.UriHelper.getUriOutputStream
import java.io.* import java.io.*
import java.util.* import java.util.*
@ -68,9 +64,6 @@ open class Database {
private var mDatabaseKDB: DatabaseKDB? = null private var mDatabaseKDB: DatabaseKDB? = null
private var mDatabaseKDBX: DatabaseKDBX? = null private var mDatabaseKDBX: DatabaseKDBX? = null
var fileUri: Uri? = null
private set
private var mSearchHelper: SearchHelper = SearchHelper() private var mSearchHelper: SearchHelper = SearchHelper()
var isReadOnly = false var isReadOnly = false
@ -548,22 +541,19 @@ open class Database {
} }
fun createData( fun createData(
databaseUri: Uri,
databaseName: String, databaseName: String,
rootName: String, rootName: String,
templateGroupName: String? templateGroupName: String?
) { ) {
setDatabaseKDBX(DatabaseKDBX(databaseName, rootName, templateGroupName)) setDatabaseKDBX(DatabaseKDBX(databaseName, rootName, templateGroupName))
this.fileUri = databaseUri
// Set Database state // Set Database state
this.dataModifiedSinceLastLoading = false this.dataModifiedSinceLastLoading = false
} }
@Throws(DatabaseInputException::class) @Throws(DatabaseInputException::class)
fun loadData( fun loadData(
contentResolver: ContentResolver, databaseStream: InputStream,
databaseUri: Uri, masterCredential: MasterCredential,
mainCredential: MainCredential,
challengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray, challengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray,
readOnly: Boolean, readOnly: Boolean,
cacheDirectory: File, cacheDirectory: File,
@ -571,16 +561,12 @@ open class Database {
fixDuplicateUUID: Boolean, fixDuplicateUUID: Boolean,
progressTaskUpdater: ProgressTaskUpdater? progressTaskUpdater: ProgressTaskUpdater?
) { ) {
// Save database URI
this.fileUri = databaseUri
// Check if the file is writable // Check if the file is writable
this.isReadOnly = readOnly this.isReadOnly = readOnly
try { try {
// Read database stream for the first time // Read database stream for the first time
readDatabaseStream(contentResolver, databaseUri, readDatabaseStream(databaseStream,
{ databaseInputStream -> { databaseInputStream ->
val databaseKDB = DatabaseKDB().apply { val databaseKDB = DatabaseKDB().apply {
binaryCache.cacheDirectory = cacheDirectory binaryCache.cacheDirectory = cacheDirectory
@ -591,8 +577,7 @@ open class Database {
progressTaskUpdater progressTaskUpdater
) { ) {
databaseKDB.deriveMasterKey( databaseKDB.deriveMasterKey(
contentResolver, masterCredential
mainCredential
) )
} }
setDatabaseKDB(databaseKDB) setDatabaseKDB(databaseKDB)
@ -607,8 +592,7 @@ open class Database {
openDatabase(databaseInputStream, openDatabase(databaseInputStream,
progressTaskUpdater) { progressTaskUpdater) {
databaseKDBX.deriveMasterKey( databaseKDBX.deriveMasterKey(
contentResolver, masterCredential,
mainCredential,
challengeResponseRetriever challengeResponseRetriever
) )
} }
@ -633,9 +617,8 @@ open class Database {
@Throws(DatabaseInputException::class) @Throws(DatabaseInputException::class)
fun mergeData( fun mergeData(
contentResolver: ContentResolver, databaseToMergeStream: InputStream,
databaseToMergeUri: Uri?, databaseToMergeMasterCredential: MasterCredential?,
databaseToMergeMainCredential: MainCredential?,
databaseToMergeChallengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray, databaseToMergeChallengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray,
isRAMSufficient: (memoryWanted: Long) -> Boolean, isRAMSufficient: (memoryWanted: Long) -> Boolean,
progressTaskUpdater: ProgressTaskUpdater? progressTaskUpdater: ProgressTaskUpdater?
@ -647,20 +630,15 @@ open class Database {
// New database instance to get new changes // New database instance to get new changes
val databaseToMerge = Database() val databaseToMerge = Database()
databaseToMerge.fileUri = databaseToMergeUri ?: this.fileUri
try { try {
val databaseUri = databaseToMerge.fileUri readDatabaseStream(databaseToMergeStream,
if (databaseUri != null) {
readDatabaseStream(contentResolver, databaseUri,
{ databaseInputStream -> { databaseInputStream ->
val databaseToMergeKDB = DatabaseKDB() val databaseToMergeKDB = DatabaseKDB()
DatabaseInputKDB(databaseToMergeKDB) DatabaseInputKDB(databaseToMergeKDB)
.openDatabase(databaseInputStream, progressTaskUpdater) { .openDatabase(databaseInputStream, progressTaskUpdater) {
if (databaseToMergeMainCredential != null) { if (databaseToMergeMasterCredential != null) {
databaseToMergeKDB.deriveMasterKey( databaseToMergeKDB.deriveMasterKey(
contentResolver, databaseToMergeMasterCredential
databaseToMergeMainCredential
) )
} else { } else {
this@Database.mDatabaseKDB?.let { thisDatabaseKDB -> this@Database.mDatabaseKDB?.let { thisDatabaseKDB ->
@ -675,10 +653,9 @@ open class Database {
DatabaseInputKDBX(databaseToMergeKDBX).apply { DatabaseInputKDBX(databaseToMergeKDBX).apply {
setMethodToCheckIfRAMIsSufficient(isRAMSufficient) setMethodToCheckIfRAMIsSufficient(isRAMSufficient)
openDatabase(databaseInputStream, progressTaskUpdater) { openDatabase(databaseInputStream, progressTaskUpdater) {
if (databaseToMergeMainCredential != null) { if (databaseToMergeMasterCredential != null) {
databaseToMergeKDBX.deriveMasterKey( databaseToMergeKDBX.deriveMasterKey(
contentResolver, databaseToMergeMasterCredential,
databaseToMergeMainCredential,
databaseToMergeChallengeResponseRetriever databaseToMergeChallengeResponseRetriever
) )
} else { } else {
@ -699,20 +676,13 @@ open class Database {
} }
databaseToMerge.mDatabaseKDB?.let { databaseKDBToMerge -> databaseToMerge.mDatabaseKDB?.let { databaseKDBToMerge ->
databaseMerger.merge(databaseKDBToMerge) databaseMerger.merge(databaseKDBToMerge)
if (databaseToMergeUri != null) {
this.dataModifiedSinceLastLoading = true this.dataModifiedSinceLastLoading = true
} }
}
databaseToMerge.mDatabaseKDBX?.let { databaseKDBXToMerge -> databaseToMerge.mDatabaseKDBX?.let { databaseKDBXToMerge ->
databaseMerger.merge(databaseKDBXToMerge) databaseMerger.merge(databaseKDBXToMerge)
if (databaseToMergeUri != null) {
this.dataModifiedSinceLastLoading = true this.dataModifiedSinceLastLoading = true
} }
} }
}
} else {
throw UnknownDatabaseLocationException()
}
} catch (e: Exception) { } catch (e: Exception) {
Log.e(TAG, "Unable to merge the database") Log.e(TAG, "Unable to merge the database")
if (e is DatabaseException) if (e is DatabaseException)
@ -725,16 +695,13 @@ open class Database {
@Throws(DatabaseInputException::class) @Throws(DatabaseInputException::class)
fun reloadData( fun reloadData(
contentResolver: ContentResolver, databaseStream: InputStream,
isRAMSufficient: (memoryWanted: Long) -> Boolean, isRAMSufficient: (memoryWanted: Long) -> Boolean,
progressTaskUpdater: ProgressTaskUpdater? progressTaskUpdater: ProgressTaskUpdater?
) { ) {
// Retrieve the stream from the old database URI
try { try {
val oldDatabaseUri = fileUri // Retrieve the stream from the old database
if (oldDatabaseUri != null) { readDatabaseStream(databaseStream,
readDatabaseStream(contentResolver, oldDatabaseUri,
{ databaseInputStream -> { databaseInputStream ->
val databaseKDB = DatabaseKDB() val databaseKDB = DatabaseKDB()
mDatabaseKDB?.let { mDatabaseKDB?.let {
@ -765,9 +732,6 @@ open class Database {
} }
) )
loaded = true loaded = true
} else {
throw UnknownDatabaseLocationException()
}
} catch (e: Exception) { } catch (e: Exception) {
Log.e(TAG, "Unable to reload the database") Log.e(TAG, "Unable to reload the database")
if (e is DatabaseException) if (e is DatabaseException)
@ -780,16 +744,12 @@ open class Database {
@Throws(Exception::class) @Throws(Exception::class)
private fun readDatabaseStream( private fun readDatabaseStream(
contentResolver: ContentResolver, databaseStream: InputStream,
databaseUri: Uri,
openDatabaseKDB: (InputStream) -> Unit, openDatabaseKDB: (InputStream) -> Unit,
openDatabaseKDBX: (InputStream) -> Unit openDatabaseKDBX: (InputStream) -> Unit
) { ) {
try { try {
// Load Data, pass Uris as InputStreams // Load Data by InputStream
val databaseStream = contentResolver.getUriInputStream(databaseUri)
?: throw UnknownDatabaseLocationException()
BufferedInputStream(databaseStream).use { databaseInputStream -> BufferedInputStream(databaseStream).use { databaseInputStream ->
// We'll end up reading 8 bytes to identify the header. Might as well use two extra. // We'll end up reading 8 bytes to identify the header. Might as well use two extra.
@ -822,26 +782,21 @@ open class Database {
@Throws(DatabaseOutputException::class) @Throws(DatabaseOutputException::class)
fun saveData( fun saveData(
contentResolver: ContentResolver, cacheFile: File,
cacheDir: File, databaseOutputStream: OutputStream?,
databaseCopyUri: Uri?, isNewLocation: Boolean,
mainCredential: MainCredential?, masterCredential: MasterCredential?,
challengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray challengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray
) { ) {
val saveUri = databaseCopyUri ?: this.fileUri
// Build temp database file to avoid file corruption if error
val cacheFile = File(cacheDir, saveUri.hashCode().toString())
try { try {
if (saveUri != null) {
// Save in a temp memory to avoid exception // Save in a temp memory to avoid exception
cacheFile.outputStream().use { outputStream -> cacheFile.outputStream().use { outputStream ->
mDatabaseKDB?.let { databaseKDB -> mDatabaseKDB?.let { databaseKDB ->
DatabaseOutputKDB(databaseKDB).apply { DatabaseOutputKDB(databaseKDB).apply {
writeDatabase(outputStream) { writeDatabase(outputStream) {
if (mainCredential != null) { if (masterCredential != null) {
databaseKDB.deriveMasterKey( databaseKDB.deriveMasterKey(
contentResolver, masterCredential
mainCredential
) )
} else { } else {
// No master key change // No master key change
@ -852,11 +807,10 @@ open class Database {
?: mDatabaseKDBX?.let { databaseKDBX -> ?: mDatabaseKDBX?.let { databaseKDBX ->
DatabaseOutputKDBX(databaseKDBX).apply { DatabaseOutputKDBX(databaseKDBX).apply {
writeDatabase(outputStream) { writeDatabase(outputStream) {
if (mainCredential != null) { if (masterCredential != null) {
// Build new master key from MainCredential // Build new master key from MainCredential
databaseKDBX.deriveMasterKey( databaseKDBX.deriveMasterKey(
contentResolver, masterCredential,
mainCredential,
challengeResponseRetriever challengeResponseRetriever
) )
} else { } else {
@ -870,16 +824,13 @@ open class Database {
} }
} }
// Copy from the cache to the final stream // Copy from the cache to the final stream
contentResolver.getUriOutputStream(saveUri)?.use { outputStream -> databaseOutputStream?.use { outputStream ->
cacheFile.inputStream().use { inputStream -> cacheFile.inputStream().use { inputStream ->
inputStream.readAllBytes { buffer -> inputStream.readAllBytes { buffer ->
outputStream.write(buffer) outputStream.write(buffer)
} }
} }
} }
} else {
throw UnknownDatabaseLocationException()
}
} catch (e: Exception) { } catch (e: Exception) {
Log.e(TAG, "Unable to save database", e) Log.e(TAG, "Unable to save database", e)
if (e is DatabaseException) if (e is DatabaseException)
@ -892,7 +843,7 @@ open class Database {
} catch (e: Exception) { } catch (e: Exception) {
Log.e(TAG, "Cache file $cacheFile cannot be deleted", e) Log.e(TAG, "Cache file $cacheFile cannot be deleted", e)
} }
if (databaseCopyUri == null) { if (isNewLocation) {
this.dataModifiedSinceLastLoading = false this.dataModifiedSinceLastLoading = false
} }
} }
@ -1010,11 +961,10 @@ open class Database {
} }
} }
fun clearAndClose(filesDirectory: File? = null) { open fun clearAndClose(filesDirectory: File? = null) {
clearIndexesAndBinaries(filesDirectory) clearIndexesAndBinaries(filesDirectory)
this.mDatabaseKDB = null this.mDatabaseKDB = null
this.mDatabaseKDBX = null this.mDatabaseKDBX = null
this.fileUri = null
this.loaded = false this.loaded = false
} }
@ -1029,11 +979,11 @@ open class Database {
} }
} }
fun validatePasswordEncoding(mainCredential: MainCredential): Boolean { fun isValidCredential(masterCredential: MasterCredential): Boolean {
val password = mainCredential.password val password = masterCredential.password
val containsKeyFile = mainCredential.keyFileUri != null val containsKeyFile = masterCredential.keyFileData != null
return mDatabaseKDB?.validatePasswordEncoding(password, containsKeyFile) return mDatabaseKDB?.isValidCredential(password, containsKeyFile)
?: mDatabaseKDBX?.validatePasswordEncoding(password, containsKeyFile) ?: mDatabaseKDBX?.isValidCredential(password, containsKeyFile)
?: false ?: false
} }

View file

@ -18,8 +18,6 @@
*/ */
package com.kunzisoft.keepass.database.element package com.kunzisoft.keepass.database.element
import android.content.ContentResolver
import android.net.Uri
import android.os.Parcel import android.os.Parcel
import android.os.Parcelable import android.os.Parcelable
import android.util.Base64 import android.util.Base64
@ -29,8 +27,9 @@ import com.kunzisoft.keepass.database.element.database.DatabaseKDBX
import com.kunzisoft.keepass.hardware.HardwareKey import com.kunzisoft.keepass.hardware.HardwareKey
import com.kunzisoft.keepass.utils.StringUtil.removeSpaceChars import com.kunzisoft.keepass.utils.StringUtil.removeSpaceChars
import com.kunzisoft.keepass.utils.StringUtil.toHexString import com.kunzisoft.keepass.utils.StringUtil.toHexString
import com.kunzisoft.keepass.utils.UriHelper.getUriInputStream import com.kunzisoft.keepass.utils.readByteArrayCompat
import com.kunzisoft.keepass.utils.readEnum import com.kunzisoft.keepass.utils.readEnum
import com.kunzisoft.keepass.utils.writeByteArrayCompat
import com.kunzisoft.keepass.utils.writeEnum import com.kunzisoft.keepass.utils.writeEnum
import org.apache.commons.codec.binary.Hex import org.apache.commons.codec.binary.Hex
import org.w3c.dom.Node import org.w3c.dom.Node
@ -43,19 +42,19 @@ import javax.xml.XMLConstants
import javax.xml.parsers.DocumentBuilderFactory import javax.xml.parsers.DocumentBuilderFactory
import javax.xml.parsers.ParserConfigurationException import javax.xml.parsers.ParserConfigurationException
data class MainCredential(var password: String? = null, data class MasterCredential(var password: String? = null,
var keyFileUri: Uri? = null, var keyFileData: ByteArray? = null,
var hardwareKey: HardwareKey? = null): Parcelable { var hardwareKey: HardwareKey? = null): Parcelable {
constructor(parcel: Parcel) : this() { constructor(parcel: Parcel) : this() {
password = parcel.readString() password = parcel.readString()
keyFileUri = parcel.readParcelable(Uri::class.java.classLoader) keyFileData = parcel.readByteArrayCompat()
hardwareKey = parcel.readEnum<HardwareKey>() hardwareKey = parcel.readEnum<HardwareKey>()
} }
override fun writeToParcel(parcel: Parcel, flags: Int) { override fun writeToParcel(parcel: Parcel, flags: Int) {
parcel.writeString(password) parcel.writeString(password)
parcel.writeParcelable(keyFileUri, flags) parcel.writeByteArrayCompat(keyFileData)
parcel.writeEnum(hardwareKey) parcel.writeEnum(hardwareKey)
} }
@ -67,10 +66,10 @@ data class MainCredential(var password: String? = null,
if (this === other) return true if (this === other) return true
if (javaClass != other?.javaClass) return false if (javaClass != other?.javaClass) return false
other as MainCredential other as MasterCredential
if (password != other.password) return false if (password != other.password) return false
if (keyFileUri != other.keyFileUri) return false if (!keyFileData.contentEquals(other.keyFileData)) return false
if (hardwareKey != other.hardwareKey) return false if (hardwareKey != other.hardwareKey) return false
return true return true
@ -78,21 +77,21 @@ data class MainCredential(var password: String? = null,
override fun hashCode(): Int { override fun hashCode(): Int {
var result = password?.hashCode() ?: 0 var result = password?.hashCode() ?: 0
result = 31 * result + (keyFileUri?.hashCode() ?: 0) result = 31 * result + (keyFileData?.hashCode() ?: 0)
result = 31 * result + (hardwareKey?.hashCode() ?: 0) result = 31 * result + (hardwareKey?.hashCode() ?: 0)
return result return result
} }
companion object CREATOR : Parcelable.Creator<MainCredential> { companion object CREATOR : Parcelable.Creator<MasterCredential> {
override fun createFromParcel(parcel: Parcel): MainCredential { override fun createFromParcel(parcel: Parcel): MasterCredential {
return MainCredential(parcel) return MasterCredential(parcel)
} }
override fun newArray(size: Int): Array<MainCredential?> { override fun newArray(size: Int): Array<MasterCredential?> {
return arrayOfNulls(size) return arrayOfNulls(size)
} }
private val TAG = MainCredential::class.java.simpleName private val TAG = MasterCredential::class.java.simpleName
@Throws(IOException::class) @Throws(IOException::class)
fun retrievePasswordKey(key: String, fun retrievePasswordKey(key: String,
@ -107,17 +106,14 @@ data class MainCredential(var password: String? = null,
} }
@Throws(IOException::class) @Throws(IOException::class)
fun retrieveFileKey(contentResolver: ContentResolver, fun retrieveKeyFileDecodedKey(
keyFileUri: Uri?, keyFileData: ByteArray,
allowXML: Boolean): ByteArray { allowXML: Boolean
if (keyFileUri == null) ): ByteArray {
throw IOException("Keyfile URI is null")
val keyData = getKeyFileData(contentResolver, keyFileUri)
?: throw IOException("No data retrieved")
try { try {
// Check XML key file // Check XML key file
val xmlKeyByteArray = if (allowXML) val xmlKeyByteArray = if (allowXML)
loadXmlKeyFile(ByteArrayInputStream(keyData)) loadXmlKeyFile(ByteArrayInputStream(keyFileData))
else else
null null
if (xmlKeyByteArray != null) { if (xmlKeyByteArray != null) {
@ -125,16 +121,16 @@ data class MainCredential(var password: String? = null,
} }
// Check 32 bytes key file // Check 32 bytes key file
when (keyData.size) { when (keyFileData.size) {
32 -> return keyData 32 -> return keyFileData
64 -> try { 64 -> try {
return Hex.decodeHex(String(keyData).toCharArray()) return Hex.decodeHex(String(keyFileData).toCharArray())
} catch (ignoredException: Exception) { } catch (ignoredException: Exception) {
// Key is not base 64, treat it as binary data // Key is not base 64, treat it as binary data
} }
} }
// Hash file as binary data // Hash file as binary data
return HashManager.hashSha256(keyData) return HashManager.hashSha256(keyFileData)
} catch (e: Exception) { } catch (e: Exception) {
throw IOException("Unable to load the keyfile.", e) throw IOException("Unable to load the keyfile.", e)
} }
@ -145,15 +141,6 @@ data class MainCredential(var password: String? = null,
return HashManager.hashSha256(keyData) return HashManager.hashSha256(keyData)
} }
@Throws(Exception::class)
private fun getKeyFileData(contentResolver: ContentResolver,
keyFileUri: Uri): ByteArray? {
contentResolver.getUriInputStream(keyFileUri)?.use { keyFileInputStream ->
return keyFileInputStream.readBytes()
}
return null
}
private fun loadXmlKeyFile(keyInputStream: InputStream): ByteArray? { private fun loadXmlKeyFile(keyInputStream: InputStream): ByteArray? {
try { try {
val documentBuilderFactory = DocumentBuilderFactory.newInstance() val documentBuilderFactory = DocumentBuilderFactory.newInstance()

View file

@ -19,13 +19,12 @@
package com.kunzisoft.keepass.database.element.database package com.kunzisoft.keepass.database.element.database
import android.content.ContentResolver
import com.kunzisoft.encrypt.HashManager import com.kunzisoft.encrypt.HashManager
import com.kunzisoft.encrypt.aes.AESTransformer import com.kunzisoft.encrypt.aes.AESTransformer
import com.kunzisoft.keepass.database.crypto.EncryptionAlgorithm import com.kunzisoft.keepass.database.crypto.EncryptionAlgorithm
import com.kunzisoft.keepass.database.crypto.kdf.KdfEngine import com.kunzisoft.keepass.database.crypto.kdf.KdfEngine
import com.kunzisoft.keepass.database.crypto.kdf.KdfFactory import com.kunzisoft.keepass.database.crypto.kdf.KdfFactory
import com.kunzisoft.keepass.database.element.MainCredential import com.kunzisoft.keepass.database.element.MasterCredential
import com.kunzisoft.keepass.database.element.binary.BinaryData import com.kunzisoft.keepass.database.element.binary.BinaryData
import com.kunzisoft.keepass.database.element.entry.EntryKDB import com.kunzisoft.keepass.database.element.entry.EntryKDB
import com.kunzisoft.keepass.database.element.group.GroupKDB import com.kunzisoft.keepass.database.element.group.GroupKDB
@ -129,25 +128,23 @@ class DatabaseKDB : DatabaseVersioned<Int, UUID, GroupKDB, EntryKDB>() {
} }
fun deriveMasterKey( fun deriveMasterKey(
contentResolver: ContentResolver, masterCredential: MasterCredential
mainCredential: MainCredential
) { ) {
// Exception when no password // Exception when no password
if (mainCredential.hardwareKey != null) if (masterCredential.hardwareKey != null)
throw HardwareKeyDatabaseException() throw HardwareKeyDatabaseException()
if (mainCredential.password == null && mainCredential.keyFileUri == null) if (masterCredential.password == null && masterCredential.keyFileData == null)
throw EmptyKeyDatabaseException() throw EmptyKeyDatabaseException()
// Retrieve plain data // Retrieve plain data
val password = mainCredential.password val password = masterCredential.password
val keyFileUri = mainCredential.keyFileUri val keyFileData = masterCredential.keyFileData
val passwordBytes = if (password != null) MainCredential.retrievePasswordKey( val passwordBytes = if (password != null) MasterCredential.retrievePasswordKey(
password, password,
passwordEncoding passwordEncoding
) else null ) else null
val keyFileBytes = if (keyFileUri != null) MainCredential.retrieveFileKey( val keyFileBytes = if (keyFileData != null) MasterCredential.retrieveKeyFileDecodedKey(
contentResolver, keyFileData,
keyFileUri,
false false
) else null ) else null

View file

@ -19,7 +19,6 @@
*/ */
package com.kunzisoft.keepass.database.element.database package com.kunzisoft.keepass.database.element.database
import android.content.ContentResolver
import android.util.Base64 import android.util.Base64
import android.util.Log import android.util.Log
import com.kunzisoft.encrypt.HashManager import com.kunzisoft.encrypt.HashManager
@ -226,24 +225,22 @@ class DatabaseKDBX : DatabaseVersioned<UUID, UUID, GroupKDBX, EntryKDBX> {
} }
fun deriveMasterKey( fun deriveMasterKey(
contentResolver: ContentResolver, masterCredential: MasterCredential,
mainCredential: MainCredential,
challengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray, challengeResponseRetriever: (HardwareKey, ByteArray?) -> ByteArray,
) { ) {
// Retrieve each plain credential // Retrieve each plain credential
val password = mainCredential.password val password = masterCredential.password
val keyFileUri = mainCredential.keyFileUri val keyFileData = masterCredential.keyFileData
val hardwareKey = mainCredential.hardwareKey val hardwareKey = masterCredential.hardwareKey
val passwordBytes = if (password != null) MainCredential.retrievePasswordKey( val passwordBytes = if (password != null) MasterCredential.retrievePasswordKey(
password, password,
passwordEncoding passwordEncoding
) else null ) else null
val keyFileBytes = if (keyFileUri != null) MainCredential.retrieveFileKey( val keyFileBytes = if (keyFileData != null) MasterCredential.retrieveKeyFileDecodedKey(
contentResolver, keyFileData,
keyFileUri,
true true
) else null ) else null
val hardwareKeyBytes = if (hardwareKey != null) MainCredential.retrieveHardwareKey( val hardwareKeyBytes = if (hardwareKey != null) MasterCredential.retrieveHardwareKey(
challengeResponseRetriever.invoke(hardwareKey, transformSeed) challengeResponseRetriever.invoke(hardwareKey, transformSeed)
) else null ) else null
@ -272,7 +269,7 @@ class DatabaseKDBX : DatabaseVersioned<UUID, UUID, GroupKDBX, EntryKDBX> {
keyFileBytes keyFileBytes
) )
} else { } else {
val hardwareKeyBytes = MainCredential.retrieveHardwareKey( val hardwareKeyBytes = MasterCredential.retrieveHardwareKey(
challengeResponseRetriever.invoke(hardwareKey, transformSeed) challengeResponseRetriever.invoke(hardwareKey, transformSeed)
) )
this.masterKey = composedKeyToMasterKey( this.masterKey = composedKeyToMasterKey(
@ -873,10 +870,10 @@ class DatabaseKDBX : DatabaseVersioned<UUID, UUID, GroupKDBX, EntryKDBX> {
} }
} }
override fun validatePasswordEncoding(password: String?, containsKeyFile: Boolean): Boolean { override fun isValidCredential(password: String?, containsKeyFile: Boolean): Boolean {
if (password == null) if (password == null)
return true return true
return super.validatePasswordEncoding(password, containsKeyFile) return super.isValidCredential(password, containsKeyFile)
} }
override fun clearIndexes() { override fun clearIndexes() {

View file

@ -93,7 +93,7 @@ abstract class DatabaseVersioned<
return null return null
} }
open fun validatePasswordEncoding(password: String?, containsKeyFile: Boolean): Boolean { open fun isValidCredential(password: String?, containsKeyFile: Boolean): Boolean {
if (password == null && !containsKeyFile) if (password == null && !containsKeyFile)
return false return false