-
-
Notifications
You must be signed in to change notification settings - Fork 1.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add Pin Quick Unlock and Remember Quick Unlock for Win/Mac #11520
base: develop
Are you sure you want to change the base?
Conversation
There is the kernel keyring, which I think the existing Linux Quick Unlock code is already using. Not sure if it can store things across sessions, though. |
Those are not persistent credential storage. They get wiped on reboot or log off. |
* Introduce QuickUnlockManager to fall back to pin unlock if OS native options are not available.
dfbec82
to
8e4cba3
Compare
Codecov ReportAttention: Patch coverage is
Additional details and impacted files@@ Coverage Diff @@
## develop #11520 +/- ##
===========================================
- Coverage 63.88% 63.81% -0.07%
===========================================
Files 362 364 +2
Lines 38147 38233 +86
===========================================
+ Hits 24368 24396 +28
- Misses 13779 13837 +58 ☔ View full report in Codecov by Sentry. |
@@ -0,0 +1,171 @@ | |||
/* | |||
* Copyright (C) 2023 KeePassXC Team <[email protected]> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
pin = QInputDialog::getText( | ||
nullptr, | ||
QObject::tr("Quick Unlock Pin Entry"), | ||
QObject::tr("Enter a %1 to %2 digit pin to use for quick unlock:").arg(MIN_PIN_LENGTH).arg(MAX_PIN_LENGTH), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There should be hyphens (%1- to %2-digit) and I guess PIN should be all-caps.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I prefer an en dash instead: 4–6 digit
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's an option, yes. But there should still be a hyphen between the number and "digit" (in addition to the en dash).
// Encrypt the data using AES-256-CBC | ||
SymmetricCipher cipher; | ||
if (!cipher.init(SymmetricCipher::Aes256_GCM, SymmetricCipher::Encrypt, key, iv)) { | ||
m_error = QObject::tr("Failed to init KeePassXC crypto."); | ||
return false; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Comment says CBC, but you're doing GCM.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Copy pasta, good catch, GCM is correct
auto pin = QInputDialog::getText( | ||
nullptr, | ||
QObject::tr("Quick Unlock Pin Entry"), | ||
QObject::tr("Enter quick unlock pin (%1 of %2 attempts):").arg(pinAttempts).arg(MAX_PIN_ATTEMPTS), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
PIN
} | ||
|
||
data.clear(); | ||
m_error = QObject::tr("Maximum pin attempts have been reached."); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
KISS: "Too many PIN attempts."
bool QuickUnlockManager::isRememberAvailable() const | ||
{ | ||
Q_UNUSED(dbUuid) | ||
if (isNativeAvailable()) { | ||
return m_nativeInterface->canRemember(); | ||
} | ||
return m_fallbackInterface->canRemember(); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like this isn't used anywhere?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yah there was a purpose at some point but the settings shuffle made this unnecessary.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This would be needed for an explicit reset if the key cannot be remembered. We should definitely do that, even if it may technically not be neccessary with the way these are implemented now. We should not make any assumptions here whether an explicit reset is necessary or not.
QByteArray encrypted = data; | ||
if (!cipher.finish(encrypted)) { | ||
m_error = QObject::tr("Failed to encrypt key data."); | ||
return false; | ||
} | ||
|
||
// Prepend the IV to the encrypted data | ||
encrypted.prepend(iv); | ||
// Store the encrypted data and pin attempts | ||
m_encryptedKeys.insert(dbUuid, qMakePair(1, encrypted)); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should probably pad the serialised key to a specific length to avoid leaking length information. I'm in general only half happy with this whole thing, since the PIN attempt count matters only as long as KeePassXC stays in full control of the cipher text (at which point it might as well not be encrypted at all).
On the other hand, the entire database is in memory naked at this point, so none of this really matters probably.
At the very least, this map should be mlocked. All this applies to Polkit unlock as well.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hmmm good idea, should maybe apply Argon2 to the pin hash to provide protection against brute force. Can just prepend the Argon2 parameters to the encrypted string.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I was thinking the same, though with a numerical four-digit PIN, Argon2 would add only very little security.
auto key = hash.result(); | ||
|
||
// Read the previously used challenge and encrypted data | ||
auto ivSize = SymmetricCipher::defaultIvSize(SymmetricCipher::Aes256_GCM); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
const auto.
@@ -0,0 +1,49 @@ | |||
/* | |||
* Copyright (C) 2023 KeePassXC Team <[email protected]> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
DatabaseOpenWidget::~DatabaseOpenWidget() | ||
{ | ||
// Reset quick unlock if we are not remembering it | ||
if (!config()->get(Config::Security_QuickUnlockRemember).toBool()) { | ||
resetQuickUnlock(); | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Also reset if remember isn't available (see above).
<property name="text"> | ||
<string>Remember quick unlock after database is closed (Touch ID / Windows Hello only)</string> | ||
</property> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Remove the part in parentheses and show this option conditionally only when either of the two is available.
I think the unlock screen layout got screwed up. There's way too much padding above "I have a key file". I also don't like the checkbox placement next to the Unlock button (why do we need that anyway, what's wrong with the way it was before?) |
I did tweak the layout of the unlock dialog, extra padding was on purpose, I felt it was too tight before. We can debate it. The checkbox is necessary now so you can decide when to use Quick Unlock before unlocking. This matters on Linux and Mac where the user isn't prompted prior to setting up quick unlock (and also potentially remembering it). On Windows or when using PIN this doesn't matter because you can always cancel the Windows Hello or PIN dialog. I am also open to changing the "Unlock" button to include an alternative drop down option of "Unlock Once" or "No Quick Unlock" or similar. |
The previous layout was deliberate. With the extra padding you're messing with the proximity and common region principles. The visual cohesion of related elements gets lost entirely and the element grouping looks loose and confusing. I'm strongly in favour of reverting the change. I don't see why we need a checkbox here. If users don't want quick unlock, they can disable it in the settings. If they don't want quick unlock just now (probably a niche use case), a hybrid button dropdown could be an option. On the other hand, you could also just lock the database and then one-click cancel quick unlock (that said: why not add that as an option to the lock button instead of the unlock button?). |
I changed the application setting from "Enable Quick Unlock" to "Use Quick Unlock by default". Enabling the new setting checks the box when you open a database, otherwise the box is unchecked. Disabling quick unlock at the application level doesn't really make sense anymore. Are you disabling the OS version of quick unlock, fallback? What if you want to use it from some databases but not others? Most users want to use quick unlock and want it to be remembered (based on issue reports and feature requests). Perhaps a compromise would be moving this checkbox to the database settings page and store the desire in database custom data that can be queried right after the database is unlocked before quick unlock is established? Then we can also potentially use a selection box: "Enable Quick Unlock (default)", "Use PIN unlock", "Disable Quick Unlock" Some people hate OS integration so explicitly allowing PIN unlock could be a neat feature for them. |
I believe in critical environments you definitely don't want to have quick unlock. So there should be an easily-accessible feature to turn it off. It may make sense to have that on database level, but I would assume that more often than not the desire to turn it off depends more on the environment than on the database (e.g. have it on at home, but not on the laptop). |
Not a fan of that. "Use PIN unlock" doesn't make any sense to me. You're not unlocking with a PIN here. Only the next unlock will use a PIN. "Disable quick unlock" sounds like a permenant setting, but isn't. I'd rather add an option to the "lock" button to cancel quick unlock immediately. Arguably, that would still store the key temporarily in the system key chain, but that's probably not the biggest deal. |
Yes, I just checked a snapshot build. The padding was indeed increased there. |
Correct me if I'm wrong but encrypting the memory with a pin hash seems like security theater. Even with 8 digits and a slow hash function this can be brute forced so quickly it's hardly an obstacle for an attacker who is able to read the memory contents. Hardware backed credential storage with Windows Hello or TouchID provide meaningful protection for the key material so they're a proper substitute for constantly filling in the passphrase, whereas the pin based feature is really just a UI level protection which can already be achieved by locking your device when leaving it unattended. I would at the very least make it a completely separate option from the hardware backed options and make it very clear to the user that it's purely a UI level protection. But without hardware backed storage options, I don't see the point to be honest. The pin pretends to be a middle ground between having to type in the passphrase and having it unlocked while in reality it doesn't provide meaningful protection and leaving it unlocked is much more convenient. |
Closes #7020
Added ability to remember quick unlock across KeePassXC sessions and database open/close. This relies on the public UUID feature introduced earlier in the develop branch to store the encrypted database credentials into the Windows Credential Store or the macOS Key Store. Remembering quick unlock is not supported on Linux due to the lack of a unified/standardized key store across distros.
Also introduce pin unlock as a fallback option when OS Native security tools are not available for use. The pin is set on first unlock and must be between 4 and 8 digits. The pin hash is used in conjunction with a random IV to encrypt database credentials in memory.
Shifted security setting to be an enable/disable quick unlock by default setting. Now that quick unlock is always available (via Pin fallback) it makes more sense to allow the user to choose what to do at time of unlock. The remember quick unlock setting is hidden on Linux since it is not functional.
Screenshots
Testing strategy
Tested on Windows, will be testing on other platforms to ensure no regression
Type of change