mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-04-04 13:07:38 +03:00
Fix OPVault import when there are multiple OTP fields
* Fix #8371 - store multiple OTP fields as `otp_#` instead of silently discarding them.
This commit is contained in:
parent
bdeef63fe4
commit
9e81c31e5a
2 changed files with 20 additions and 1 deletions
|
@ -84,7 +84,16 @@ void OpVaultReader::fillFromSectionField(Entry* entry, const QString& sectionNam
|
||||||
auto kind = field["k"].toString();
|
auto kind = field["k"].toString();
|
||||||
|
|
||||||
if (attrName.startsWith("TOTP_")) {
|
if (attrName.startsWith("TOTP_")) {
|
||||||
if (attrValue.startsWith("otpauth://")) {
|
if (entry->hasTotp()) {
|
||||||
|
// Store multiple TOTP definitions as additional otp attributes
|
||||||
|
int i = 0;
|
||||||
|
QString name("otp");
|
||||||
|
auto attributes = entry->attributes()->keys();
|
||||||
|
while (attributes.contains(name)) {
|
||||||
|
name = QString("otp_%1").arg(++i);
|
||||||
|
}
|
||||||
|
entry->attributes()->set(name, attrValue);
|
||||||
|
} else if (attrValue.startsWith("otpauth://")) {
|
||||||
QUrlQuery query(attrValue);
|
QUrlQuery query(attrValue);
|
||||||
// at least as of 1Password 7, they don't append the digits= and period= which totp.cpp requires
|
// at least as of 1Password 7, they don't append the digits= and period= which totp.cpp requires
|
||||||
if (!query.hasQueryItem("digits")) {
|
if (!query.hasQueryItem("digits")) {
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "format/OpVaultReader.h"
|
#include "format/OpVaultReader.h"
|
||||||
#include "totp/totp.h"
|
#include "totp/totp.h"
|
||||||
|
|
||||||
|
#include <QJsonObject>
|
||||||
#include <QList>
|
#include <QList>
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
#include <QTest>
|
#include <QTest>
|
||||||
|
@ -110,6 +111,15 @@ void TestOpVaultReader::testReadIntoDatabase()
|
||||||
QCOMPARE(totpSettings->digits, static_cast<unsigned int>(8));
|
QCOMPARE(totpSettings->digits, static_cast<unsigned int>(8));
|
||||||
QCOMPARE(totpSettings->step, static_cast<unsigned int>(45));
|
QCOMPARE(totpSettings->step, static_cast<unsigned int>(45));
|
||||||
|
|
||||||
|
// Add another OTP to this entry to confirm it doesn't overwrite the existing one
|
||||||
|
auto field = QJsonObject::fromVariantMap({{"n", "TOTP_SETTINGS"}, {"v", "otpauth://test.url?digits=6"}});
|
||||||
|
reader.fillFromSectionField(entry, "", field);
|
||||||
|
QVERIFY(entry->hasTotp());
|
||||||
|
totpSettings = entry->totpSettings();
|
||||||
|
QCOMPARE(totpSettings->digits, static_cast<unsigned int>(8));
|
||||||
|
QCOMPARE(totpSettings->step, static_cast<unsigned int>(45));
|
||||||
|
QVERIFY(entry->attributes()->contains("otp_1"));
|
||||||
|
|
||||||
// Confirm trashed entries are sent to the recycle bin
|
// Confirm trashed entries are sent to the recycle bin
|
||||||
auto recycleBin = db->metadata()->recycleBin();
|
auto recycleBin = db->metadata()->recycleBin();
|
||||||
QVERIFY(recycleBin);
|
QVERIFY(recycleBin);
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue